pkgmaker 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315
  1. unset MAKEFLAGS
  2. export MAKEFLAGS=s
  3. cd "$(dirname "$0")"
  4. export TOPDIR=$(realpath ..)
  5. if gmake --help >/dev/null 2>&1; then
  6. export GMAKE=gmake
  7. else
  8. export GMAKE=make
  9. fi
  10. GMAKE="$GMAKE --no-print-directory"
  11. (( x_cols = (COLUMNS > 10) ? COLUMNS - 2 : 80 ))
  12. typeset -L$x_cols pbar
  13. # build a cache of “ipkg package name” → “package conf option” for
  14. # use with dependency resolution
  15. rm -rf pkglist.d
  16. mkdir pkglist.d
  17. for dn in */Makefile; do
  18. dn=${dn%/*}
  19. pbar="Pass 1: $dn ..."
  20. print -nu2 "$pbar\r"
  21. cd $dn
  22. # ALL_PKGOPTS: all subpackage conf options
  23. # PKGNAME_*: subpackage (ipkg) package name, by subpackage option
  24. eval $($GMAKE dump="ALL_PKGOPTS \
  25. \$(foreach x,\${ALL_PKGOPTS},PKGNAME_\${x})")
  26. cd ..
  27. if [[ -z $ALL_PKGOPTS ]]; then
  28. #print -u2 "Warning: $dn/Makefile contains no packages, skipped"
  29. continue
  30. fi
  31. for spcu in $ALL_PKGOPTS; do # spcu: package option, ucase
  32. eval sppn=\$PKGNAME_$spcu # sppn: subpackage (ipkg) name
  33. # once mksh R40 is out, use its new associative arrays here!
  34. print -r -- "$spcu" >pkglist.d/"$sppn"
  35. done
  36. done
  37. # build Config.in files and resolve dependencies
  38. rm -f package_sections
  39. rm -rf pkgconfigs.d
  40. mkdir pkgconfigs.d
  41. for dn in */Makefile; do
  42. dn=${dn%/*}
  43. pbar="Pass 2: $dn ..."
  44. print -nu2 "$pbar\r"
  45. mkdir pkgconfigs.d/$dn
  46. cd $dn
  47. # PKG_NAME: package name (directory, free-format)
  48. # PKG_FLAVOURS: all package flavours (boolean options), uppercase
  49. # PKG_DESCR: package description (directory)
  50. # PKG_URL: package homepage
  51. # PKG_CXX: uppercase varname part to use for CFrustFrust checks
  52. # ALL_PKGOPTS: all subpackage conf options
  53. # PKGNAME_*: subpackage (ipkg) package name, by subpackage option
  54. # PKGDESC_*: subpackage description, by subpackage option
  55. # PKGDEPS_*: subpackage depends on ipkg packages, by subpkg option
  56. # PKGDFLT_*: subpackage 'default {:-n}', by subpackage option
  57. # CFLINE_*: one free-format Config.in line per subpackage option
  58. # PKGFD_*: flavour description, per package flavour option
  59. # PKG_{HOST,TARGET}_DEPENDS: add host or target dependencies
  60. eval $($GMAKE dump="PKG_NAME PKG_FLAVOURS PKG_DESCR PKG_SECTION PKG_URL PKG_MULTI PKG_CXX \
  61. ALL_PKGOPTS \$(foreach x,\${ALL_PKGOPTS},PKGNAME_\${x} \
  62. PKGDESC_\${x} PKGDEPS_\${x} PKGDFLT_\${x} PKGSECT_\${x} CFLINE_\${x}) \
  63. \$(foreach x,\${PKG_FLAVOURS},PKGFD_\${x}) \
  64. PKG_HOST_DEPENDS PKG_TARGET_DEPENDS")
  65. # dnu: directory name, uppercase, y/-+/_X/
  66. typeset -u dnu=${dn//-/_}
  67. dnu=${dnu//+/X}
  68. echo "pkgconfigs.d/$dn/Config.in ${PKG_SECTION:=none}" >>../package_sections
  69. echo "$dn/Config.in.manual ${PKG_SECTION:=none}" >>../package_sections
  70. # skip if we take care of this one manually
  71. [[ $dn != "base-files" ]] && [[ -s Config.in.manual ]] && { cd ..; continue; }
  72. exec 4>../pkgconfigs.d/$dn/Config.in
  73. # Handle master package (directory)
  74. print -u4 "config ADK_COMPILE_$dnu"
  75. if [[ -z $ALL_PKGOPTS ]]; then
  76. # pseudo package, does not produce an ipkg package
  77. ppnf=$PKG_NAME # ppnf: pseudopkg name, filled
  78. if [[ -n $PKG_DESCR ]]; then
  79. while (( ${#ppnf} < 34 )); do
  80. ppnf=$ppnf.
  81. done
  82. ppnf="$ppnf $PKG_DESCR"
  83. fi
  84. print -u4 "\tprompt \"$ppnf\""
  85. fi
  86. print -u4 \\ttristate
  87. if [[ -n $ALL_PKGOPTS ]]; then
  88. # real (master) package, contains 1+ ipkg (sub)packages
  89. print -nu4 \\tdepends on
  90. sp=' ' # local sp: space (or ' || ')
  91. for spcu in $ALL_PKGOPTS; do # spcu: package option, ucase
  92. if [[ -n $PKG_MULTI ]]; then
  93. if [[ $dnu != $spcu ]]; then
  94. print -nu4 "${sp}ADK_PACKAGE_$spcu"
  95. sp=' || '
  96. else
  97. print -nu4 "${sp}ADK_HAVE_DOT_CONFIG"
  98. sp=' || '
  99. fi
  100. else
  101. print -nu4 "${sp}ADK_PACKAGE_$spcu"
  102. sp=' || '
  103. fi
  104. done
  105. print -u4
  106. fi
  107. print -u4 \\tdefault n
  108. # Handle NOT/ONLY_FOR_PLATFORM alikes
  109. phd= # phd: PKG_HOST_DEPENDS expand.
  110. if [[ -n $PKG_HOST_DEPENDS ]]; then
  111. phd='\tdepends on'
  112. if [[ $PKG_HOST_DEPENDS = *\!* ]]; then
  113. sp=' !'
  114. else
  115. sp=' '
  116. fi
  117. for x in $PKG_HOST_DEPENDS; do
  118. typeset -u x=${x#!}
  119. phd="$phd${sp}ADK_HOST_$x"
  120. if [[ $PKG_HOST_DEPENDS = *\!* ]]; then
  121. sp=' && !'
  122. else
  123. sp=' || '
  124. fi
  125. done
  126. fi
  127. ptd= # ptd: PKG_TARGET_DEPENDS exp.
  128. if [[ -n $PKG_TARGET_DEPENDS ]]; then
  129. ptd='\tdepends on'
  130. sp=' ' # local sp: space (or ' || ')
  131. if [[ $PKG_TARGET_DEPENDS = *\!* ]]; then
  132. sp=' !'
  133. else
  134. sp=' '
  135. fi
  136. for x in $PKG_TARGET_DEPENDS; do
  137. typeset -l x=${x#!}
  138. #XXX cache this with mksh R40+
  139. found=0
  140. while read friendlyname sym; do
  141. [[ $friendlyname = $x ]] || continue
  142. found=1
  143. break
  144. done <../../target/target.lst
  145. if (( !found )); then
  146. print -u2 "$dn: Target '$x' not found!"
  147. exit 1
  148. fi
  149. ptd="$ptd${sp}$sym"
  150. if [[ $PKG_TARGET_DEPENDS = *\!* ]]; then
  151. sp=' && !'
  152. else
  153. sp=' || '
  154. fi
  155. done
  156. fi
  157. # Handle subpackages / multipackages
  158. for spcu in $ALL_PKGOPTS; do # spcu: package option, ucase
  159. eval sppn=\$PKGNAME_$spcu # sppn: subpackage (ipkg) name
  160. eval desc=\$PKGDESC_$spcu # desc: subpackage description
  161. : ${desc:=$PKG_DESCR} # take from main pkg if empty
  162. eval sect=\$PKGSECT_$spcu # sect: subpackage section
  163. : ${sect:=$PKG_SECTION} # take from main pkg if empty
  164. eval deps=\$PKGDEPS_$spcu # deps: subpackage dependencies
  165. eval dflt=\$PKGDFLT_$spcu # dflt: config 'default' opt.
  166. eval xline=\$CFLINE_$spcu # xline: one free-format line
  167. echo "pkgconfigs.d/$dn/Config.in.$sppn $sect" >>../package_sections
  168. exec 4>../pkgconfigs.d/$dn/Config.in.$sppn
  169. h=4
  170. print -u$h config ADK_PACKAGE_$spcu
  171. spnf=$sppn # spnf: subpackage name, filled
  172. if [[ -n ${desc:-$PKG_NAME} ]]; then
  173. while (( ${#spnf} < 34 )); do
  174. spnf=$spnf.
  175. done
  176. spnf="$spnf ${desc:-$PKG_NAME}"
  177. fi
  178. print -u$h "\tprompt \"$spnf\""
  179. print -u$h \\ttristate
  180. if [[ -n $PKG_MULTI ]]; then
  181. if [[ $spcu != $dnu ]]; then
  182. print -u$h "\tdepends on ADK_PACKAGE_$dnu"
  183. fi
  184. fi
  185. [[ -n $phd ]] && print -u$h "$phd"
  186. [[ -n $ptd ]] && print -u$h "$ptd"
  187. print -u$h "\tdefault ${dflt:-n}"
  188. for dep in $deps; do # dep: ipkg name of one rundep.
  189. # skip dependencies on uclibc++ and libstdcxx iff
  190. # we produce these automatically
  191. [[ -n $PKG_CXX && $dep = @(uclibc++|libstdcxx) ]] && \
  192. continue
  193. case $dep {
  194. (kmod-*)
  195. # produce dependency on kernel package
  196. # which have special name→sym mangling
  197. typeset -u udep=${dep//-/_}
  198. print -u$h "\tselect ADK_KPACKAGE_$udep"
  199. ;;
  200. (*)
  201. # produce dependency on regular package
  202. # where the symbol is cached (see above)
  203. print -u$h '\tselect' \
  204. ADK_PACKAGE_$(<../pkglist.d/"$dep")
  205. ;;
  206. }
  207. done
  208. print -u$h \\tselect ADK_COMPILE_$dnu
  209. [[ -n $xline ]] && print -u$h "\t$xline"
  210. if [[ -n $desc$PKG_URL ]]; then
  211. # produce (optional) help text
  212. print -u$h \\thelp
  213. [[ -n $desc ]] && print -u$h "\t $desc"
  214. [[ -n $desc && -n $PKG_URL ]] && print -u$h '\t '
  215. [[ -n $PKG_URL ]] && print -u$h "\t WWW: $PKG_URL"
  216. fi
  217. done
  218. # Handle CFrustFrust library selection, if necessary
  219. [[ -n $PKG_CXX ]] && cat >&4 <<EOF
  220. choice
  221. prompt "C++ library to use"
  222. depends on ADK_COMPILE_$dnu
  223. default ADK_COMPILE_${PKG_CXX}_WITH_STDCXX if ADK_TARGET_LIB_GLIBC || ADK_TARGET_LIB_EGLIBC
  224. default ADK_COMPILE_${PKG_CXX}_WITH_UCLIBCXX if ADK_TARGET_LIB_UCLIBC
  225. config ADK_COMPILE_${PKG_CXX}_WITH_STDCXX
  226. bool "GNU C++ library"
  227. select ADK_PACKAGE_LIBSTDCXX
  228. config ADK_COMPILE_${PKG_CXX}_WITH_UCLIBCXX
  229. bool "uClibc++ library"
  230. select ADK_PACKAGE_UCLIBCXX
  231. endchoice
  232. EOF
  233. # Handle flavours (per directory)
  234. for pfcu in $PKG_FLAVOURS; do # pfcu: pkg flavour conf opt.
  235. eval pfd=\$PKGFD_$pfcu # pfd: pkg flavour description
  236. print
  237. print config ADK_PACKAGE_${dnu}_$pfcu
  238. print "\tbool \"${pfd:-$PKG_NAME -> flavour $pfcu}\""
  239. print \\tdefault n
  240. print \\tdepends on ADK_COMPILE_$dnu
  241. print \\thelp
  242. print "\t flavour ADK_PACKAGE_${dnu}_$pfcu for $PKG_NAME"
  243. done >&4
  244. cd ..
  245. done
  246. # return good if given file exists and is non-empty
  247. function non_empty_file() {
  248. [[ -f "$1" ]] || return 1
  249. [[ -n "$(cat "$1")" ]] || return 1
  250. return 0
  251. }
  252. # print the verbose section name for a given section tag
  253. function lookup_section_string() {
  254. str="$(grep ^$1\ SECTIONS.list | cut -d ' ' -f '2-')"
  255. [[ -n $str ]] && { echo $str; return; }
  256. echo $1
  257. }
  258. # print the first prompt's first word's value in a given Config.in file
  259. function get_first_prompt() {
  260. prompt="$(grep -m 1 "prompt " $1 | sed -n 's/.*"\([^ \.]*\)[ \.].*"/\1/p')"
  261. [[ -n $prompt ]] && echo $prompt
  262. }
  263. # prepare Config.in list for sorting
  264. while read config_in section; do
  265. pbar="Pass 3: $config_in ..."
  266. print -nu2 "$pbar\r"
  267. non_empty_file $config_in || continue
  268. prompt="$(get_first_prompt $config_in)"
  269. [[ -n $prompt ]] || continue
  270. echo "$prompt $config_in $(lookup_section_string $section)"
  271. done <package_sections >package_section_list
  272. # create the Config.in.auto from the sorted list from above
  273. cursec=""
  274. sort -k 3 -k 1 -f package_section_list | while read name file section; do
  275. pbar="Pass 4: $name ..."
  276. print -nu2 "$pbar\r"
  277. if [[ $cursec != $section ]]; then
  278. [[ -n $cursec ]] && print "endmenu\n"
  279. print "menu \"$section\""
  280. cursec="$section"
  281. fi
  282. print "source \"package/$file\""
  283. done >Config.in.auto
  284. print "endmenu\n" >>Config.in.auto
  285. rm -f package_sections package_section_list