package_rpm.bbclass 32 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776
  1. #
  2. # Copyright OpenEmbedded Contributors
  3. #
  4. # SPDX-License-Identifier: MIT
  5. #
  6. inherit package
  7. IMAGE_PKGTYPE ?= "rpm"
  8. RPM = "rpm"
  9. RPMBUILD = "rpmbuild"
  10. RPMBUILD_COMPMODE ?= "${@'w%dT%d.zstdio' % (int(d.getVar('ZSTD_COMPRESSION_LEVEL')), int(d.getVar('ZSTD_THREADS')))}"
  11. PKGWRITEDIRRPM = "${WORKDIR}/deploy-rpms"
  12. # Maintaining the perfile dependencies has significant overhead when writing the
  13. # packages. When set, this value merges them for efficiency.
  14. MERGEPERFILEDEPS = "1"
  15. # Filter dependencies based on a provided function.
  16. def filter_deps(var, f):
  17. import collections
  18. depends_dict = bb.utils.explode_dep_versions2(var)
  19. newdeps_dict = collections.OrderedDict()
  20. for dep in depends_dict:
  21. if f(dep):
  22. newdeps_dict[dep] = depends_dict[dep]
  23. return bb.utils.join_deps(newdeps_dict, commasep=False)
  24. # Filter out absolute paths (typically /bin/sh and /usr/bin/env) and any perl
  25. # dependencies for nativesdk packages.
  26. def filter_nativesdk_deps(srcname, var):
  27. if var and srcname.startswith("nativesdk-"):
  28. var = filter_deps(var, lambda dep: not dep.startswith('/') and dep != 'perl' and not dep.startswith('perl('))
  29. return var
  30. # Construct per file dependencies file
  31. def write_rpm_perfiledata(srcname, d):
  32. import oe.package
  33. workdir = d.getVar('WORKDIR')
  34. packages = d.getVar('PACKAGES')
  35. pkgd = d.getVar('PKGD')
  36. def dump_filerdeps(varname, outfile, d):
  37. outfile.write("#!/usr/bin/env python3\n\n")
  38. outfile.write("# Dependency table\n")
  39. outfile.write('deps = {\n')
  40. for pkg in packages.split():
  41. dependsflist_key = 'FILE' + varname + 'FLIST' + ":" + pkg
  42. dependsflist = (d.getVar(dependsflist_key) or "")
  43. for dfile in dependsflist.split():
  44. key = "FILE" + varname + ":" + dfile + ":" + pkg
  45. deps = filter_nativesdk_deps(srcname, d.getVar(key) or "")
  46. depends_dict = bb.utils.explode_dep_versions(deps)
  47. file = oe.package.file_reverse_translate(dfile)
  48. outfile.write('"' + pkgd + file + '" : "')
  49. for dep in depends_dict:
  50. ver = depends_dict[dep]
  51. if dep and ver:
  52. ver = ver.replace("(", "")
  53. ver = ver.replace(")", "")
  54. outfile.write(dep + " " + ver + " ")
  55. else:
  56. outfile.write(dep + " ")
  57. outfile.write('",\n')
  58. outfile.write('}\n\n')
  59. outfile.write("import sys\n")
  60. outfile.write("while 1:\n")
  61. outfile.write("\tline = sys.stdin.readline().strip()\n")
  62. outfile.write("\tif not line:\n")
  63. outfile.write("\t\tsys.exit(0)\n")
  64. outfile.write("\tif line in deps:\n")
  65. outfile.write("\t\tprint(deps[line] + '\\n')\n")
  66. # OE-core dependencies a.k.a. RPM requires
  67. outdepends = workdir + "/" + srcname + ".requires"
  68. dependsfile = open(outdepends, 'w')
  69. dump_filerdeps('RDEPENDS', dependsfile, d)
  70. dependsfile.close()
  71. os.chmod(outdepends, 0o755)
  72. # OE-core / RPM Provides
  73. outprovides = workdir + "/" + srcname + ".provides"
  74. providesfile = open(outprovides, 'w')
  75. dump_filerdeps('RPROVIDES', providesfile, d)
  76. providesfile.close()
  77. os.chmod(outprovides, 0o755)
  78. return (outdepends, outprovides)
  79. python write_specfile () {
  80. import oe.packagedata
  81. import os,pwd,grp,stat
  82. # append information for logs and patches to %prep
  83. def add_prep(d, spec_files_bottom):
  84. if d.getVarFlag('ARCHIVER_MODE', 'srpm') == '1' and bb.data.inherits_class('archiver', d):
  85. spec_files_bottom.append('%%prep')
  86. spec_files_bottom.append('%s' % "echo \"include logs and patches, Please check them in SOURCES\"")
  87. spec_files_bottom.append('')
  88. # append the name of tarball to key word 'SOURCE' in xxx.spec.
  89. def tail_source(d):
  90. if d.getVarFlag('ARCHIVER_MODE', 'srpm') == '1' and bb.data.inherits_class('archiver', d):
  91. ar_outdir = d.getVar('ARCHIVER_OUTDIR')
  92. if not os.path.exists(ar_outdir):
  93. return
  94. source_list = os.listdir(ar_outdir)
  95. source_number = 0
  96. for source in source_list:
  97. # do_deploy_archives may have already run (from sstate) meaning a .src.rpm may already
  98. # exist in ARCHIVER_OUTDIR so skip if present.
  99. if source.endswith(".src.rpm"):
  100. continue
  101. # The rpmbuild doesn't need the root permission, but it needs
  102. # to know the file's user and group name, the only user and
  103. # group in fakeroot is "root" when working in fakeroot.
  104. f = os.path.join(ar_outdir, source)
  105. os.chown(f, 0, 0)
  106. spec_preamble_top.append('Source%s: %s' % (source_number, source))
  107. source_number += 1
  108. # In RPM, dependencies are of the format: pkg <>= Epoch:Version-Release
  109. # This format is similar to OE, however there are restrictions on the
  110. # characters that can be in a field. In the Version field, "-"
  111. # characters are not allowed. "-" is allowed in the Release field.
  112. #
  113. # We translate the "-" in the version to a "+", by loading the PKGV
  114. # from the dependent recipe, replacing the - with a +, and then using
  115. # that value to do a replace inside of this recipe's dependencies.
  116. # This preserves the "-" separator between the version and release, as
  117. # well as any "-" characters inside of the release field.
  118. #
  119. # All of this has to happen BEFORE the mapping_rename_hook as
  120. # after renaming we cannot look up the dependencies in the packagedata
  121. # store.
  122. def translate_vers(varname, d):
  123. depends = d.getVar(varname)
  124. if depends:
  125. depends_dict = bb.utils.explode_dep_versions2(depends)
  126. newdeps_dict = {}
  127. for dep in depends_dict:
  128. verlist = []
  129. for ver in depends_dict[dep]:
  130. if '-' in ver:
  131. subd = oe.packagedata.read_subpkgdata_dict(dep, d)
  132. if 'PKGV' in subd:
  133. pv = subd['PV']
  134. pkgv = subd['PKGV']
  135. reppv = pkgv.replace('-', '+')
  136. if ver.startswith(pv):
  137. ver = ver.replace(pv, reppv)
  138. ver = ver.replace(pkgv, reppv)
  139. if 'PKGR' in subd:
  140. # Make sure PKGR rather than PR in ver
  141. pr = '-' + subd['PR']
  142. pkgr = '-' + subd['PKGR']
  143. if pkgr not in ver:
  144. ver = ver.replace(pr, pkgr)
  145. verlist.append(ver)
  146. else:
  147. verlist.append(ver)
  148. newdeps_dict[dep] = verlist
  149. depends = bb.utils.join_deps(newdeps_dict)
  150. d.setVar(varname, depends.strip())
  151. # We need to change the style the dependency from BB to RPM
  152. # This needs to happen AFTER the mapping_rename_hook
  153. def print_deps(variable, tag, array, d):
  154. depends = variable
  155. if depends:
  156. depends_dict = bb.utils.explode_dep_versions2(depends)
  157. for dep in depends_dict:
  158. for ver in depends_dict[dep]:
  159. ver = ver.replace('(', '')
  160. ver = ver.replace(')', '')
  161. array.append("%s: %s %s" % (tag, dep, ver))
  162. if not len(depends_dict[dep]):
  163. array.append("%s: %s" % (tag, dep))
  164. def walk_files(walkpath, target, conffiles):
  165. # We can race against the ipk/deb backends which create CONTROL or DEBIAN directories
  166. # when packaging. We just ignore these files which are created in
  167. # packages-split/ and not package/
  168. # We have the odd situation where the CONTROL/DEBIAN directory can be removed in the middle of
  169. # of the walk, the isdir() test would then fail and the walk code would assume its a file
  170. # hence we check for the names in files too.
  171. for rootpath, dirs, files in os.walk(walkpath):
  172. def get_attr(path):
  173. stat_f = os.stat(rootpath + "/" + path, follow_symlinks=False)
  174. mode = stat.S_IMODE(stat_f.st_mode)
  175. try:
  176. owner = pwd.getpwuid(stat_f.st_uid).pw_name
  177. except Exception as e:
  178. filename = d.getVar('RECIPE_SYSROOT') + '/etc/passwd'
  179. if os.path.exists(filename):
  180. bb.error("Content of /etc/passwd in sysroot:\n{}".format(
  181. open(filename).read()))
  182. else:
  183. bb.error("File {} doesn't exist in sysroot!".format(filename))
  184. raise e
  185. try:
  186. group = grp.getgrgid(stat_f.st_gid).gr_name
  187. except Exception as e:
  188. filename = d.getVar("RECIPE_SYSROOT") +"/etc/group"
  189. if os.path.exists(filename):
  190. bb.error("Content of /etc/group in sysroot:\n{}".format(
  191. open(filename).read()))
  192. else:
  193. bb.error("File {} doesn't exists in sysroot!".format(filename))
  194. raise e
  195. return "%attr({:o},{},{}) ".format(mode, owner, group)
  196. def escape_chars(p):
  197. return p.replace("%", "%%").replace("\\", "\\\\").replace('"', '\\"')
  198. path = rootpath.replace(walkpath, "")
  199. if path.endswith("DEBIAN") or path.endswith("CONTROL"):
  200. continue
  201. # Treat all symlinks to directories as normal files.
  202. # os.walk() lists them as directories.
  203. def move_to_files(dir):
  204. if os.path.islink(os.path.join(rootpath, dir)):
  205. files.append(dir)
  206. return True
  207. else:
  208. return False
  209. dirs[:] = [dir for dir in dirs if not move_to_files(dir)]
  210. for dir in dirs:
  211. if dir == "CONTROL" or dir == "DEBIAN":
  212. continue
  213. p = path + '/' + dir
  214. # All packages own the directories their files are in...
  215. target.append(get_attr(dir) + '%dir "' + escape_chars(p) + '"')
  216. for file in files:
  217. if file == "CONTROL" or file == "DEBIAN":
  218. continue
  219. attr = get_attr(file)
  220. p = path + '/' + file
  221. if conffiles.count(p):
  222. target.append(attr + '%config "' + escape_chars(p) + '"')
  223. else:
  224. target.append(attr + '"' + escape_chars(p) + '"')
  225. # Prevent the prerm/postrm scripts from being run during an upgrade
  226. def wrap_uninstall(scriptvar):
  227. scr = scriptvar.strip()
  228. if scr.startswith("#!"):
  229. pos = scr.find("\n") + 1
  230. else:
  231. pos = 0
  232. scr = scr[:pos] + 'if [ "$1" = "0" ] ; then\n' + scr[pos:] + '\nfi'
  233. return scr
  234. def get_perfile(varname, pkg, d):
  235. deps = []
  236. dependsflist_key = 'FILE' + varname + 'FLIST' + ":" + pkg
  237. dependsflist = (d.getVar(dependsflist_key) or "")
  238. for dfile in dependsflist.split():
  239. key = "FILE" + varname + ":" + dfile + ":" + pkg
  240. depends = d.getVar(key)
  241. if depends:
  242. deps.append(depends)
  243. return " ".join(deps)
  244. def append_description(spec_preamble, text):
  245. """
  246. Add the description to the spec file.
  247. """
  248. import textwrap
  249. dedent_text = textwrap.dedent(text).strip()
  250. # Bitbake saves "\n" as "\\n"
  251. if '\\n' in dedent_text:
  252. for t in dedent_text.split('\\n'):
  253. spec_preamble.append(t.strip())
  254. else:
  255. spec_preamble.append('%s' % textwrap.fill(dedent_text, width=75))
  256. packages = d.getVar('PACKAGES')
  257. if not packages or packages == '':
  258. bb.debug(1, "No packages; nothing to do")
  259. return
  260. pkgdest = d.getVar('PKGDEST')
  261. if not pkgdest:
  262. bb.fatal("No PKGDEST")
  263. outspecfile = d.getVar('OUTSPECFILE')
  264. if not outspecfile:
  265. bb.fatal("No OUTSPECFILE")
  266. # Construct the SPEC file...
  267. srcname = d.getVar('PN')
  268. localdata = bb.data.createCopy(d)
  269. localdata.setVar('OVERRIDES', d.getVar("OVERRIDES", False) + ":" + srcname)
  270. srcsummary = (localdata.getVar('SUMMARY') or localdata.getVar('DESCRIPTION') or ".")
  271. srcversion = localdata.getVar('PKGV').replace('-', '+')
  272. srcrelease = localdata.getVar('PKGR')
  273. srcepoch = (localdata.getVar('PKGE') or "")
  274. srclicense = localdata.getVar('LICENSE')
  275. srcsection = localdata.getVar('SECTION')
  276. srcmaintainer = localdata.getVar('MAINTAINER')
  277. srchomepage = localdata.getVar('HOMEPAGE')
  278. srcdescription = localdata.getVar('DESCRIPTION') or "."
  279. srccustomtagschunk = oe.packagedata.get_package_additional_metadata("rpm", localdata)
  280. srcdepends = d.getVar('DEPENDS')
  281. srcrdepends = ""
  282. srcrrecommends = ""
  283. srcrsuggests = ""
  284. srcrprovides = ""
  285. srcrreplaces = ""
  286. srcrconflicts = ""
  287. srcrobsoletes = ""
  288. srcrpreinst = []
  289. srcrpostinst = []
  290. srcrprerm = []
  291. srcrpostrm = []
  292. spec_preamble_top = []
  293. spec_preamble_bottom = []
  294. spec_scriptlets_top = []
  295. spec_scriptlets_bottom = []
  296. spec_files_top = []
  297. spec_files_bottom = []
  298. perfiledeps = (d.getVar("MERGEPERFILEDEPS") or "0") == "0"
  299. extra_pkgdata = (d.getVar("RPM_EXTRA_PKGDATA") or "0") == "1"
  300. for pkg in packages.split():
  301. localdata = bb.data.createCopy(d)
  302. root = "%s/%s" % (pkgdest, pkg)
  303. localdata.setVar('ROOT', '')
  304. localdata.setVar('ROOT_%s' % pkg, root)
  305. pkgname = localdata.getVar('PKG:%s' % pkg)
  306. if not pkgname:
  307. pkgname = pkg
  308. localdata.setVar('PKG', pkgname)
  309. localdata.setVar('OVERRIDES', d.getVar("OVERRIDES", False) + ":" + pkg)
  310. conffiles = oe.package.get_conffiles(pkg, d)
  311. splitname = pkgname
  312. splitsummary = (localdata.getVar('SUMMARY') or localdata.getVar('DESCRIPTION') or ".")
  313. splitversion = (localdata.getVar('PKGV') or "").replace('-', '+')
  314. splitrelease = (localdata.getVar('PKGR') or "")
  315. splitepoch = (localdata.getVar('PKGE') or "")
  316. splitlicense = (localdata.getVar('LICENSE') or "")
  317. splitsection = (localdata.getVar('SECTION') or "")
  318. splitdescription = (localdata.getVar('DESCRIPTION') or ".")
  319. splitcustomtagschunk = oe.packagedata.get_package_additional_metadata("rpm", localdata)
  320. translate_vers('RDEPENDS', localdata)
  321. translate_vers('RRECOMMENDS', localdata)
  322. translate_vers('RSUGGESTS', localdata)
  323. translate_vers('RPROVIDES', localdata)
  324. translate_vers('RREPLACES', localdata)
  325. translate_vers('RCONFLICTS', localdata)
  326. # Map the dependencies into their final form
  327. oe.packagedata.mapping_rename_hook(localdata)
  328. splitrdepends = localdata.getVar('RDEPENDS') or ""
  329. splitrrecommends = localdata.getVar('RRECOMMENDS') or ""
  330. splitrsuggests = localdata.getVar('RSUGGESTS') or ""
  331. splitrprovides = localdata.getVar('RPROVIDES') or ""
  332. splitrreplaces = localdata.getVar('RREPLACES') or ""
  333. splitrconflicts = localdata.getVar('RCONFLICTS') or ""
  334. splitrobsoletes = ""
  335. splitrpreinst = localdata.getVar('pkg_preinst')
  336. splitrpostinst = localdata.getVar('pkg_postinst')
  337. splitrprerm = localdata.getVar('pkg_prerm')
  338. splitrpostrm = localdata.getVar('pkg_postrm')
  339. if not perfiledeps:
  340. # Add in summary of per file dependencies
  341. splitrdepends = splitrdepends + " " + get_perfile('RDEPENDS', pkg, d)
  342. splitrprovides = splitrprovides + " " + get_perfile('RPROVIDES', pkg, d)
  343. splitrdepends = filter_nativesdk_deps(srcname, splitrdepends)
  344. # Gather special src/first package data
  345. if srcname == splitname:
  346. archiving = d.getVarFlag('ARCHIVER_MODE', 'srpm') == '1' and \
  347. bb.data.inherits_class('archiver', d)
  348. if archiving and srclicense != splitlicense:
  349. bb.warn("The SRPM produced may not have the correct overall source license in the License tag. This is due to the LICENSE for the primary package and SRPM conflicting.")
  350. srclicense = splitlicense
  351. srcrdepends = splitrdepends
  352. srcrrecommends = splitrrecommends
  353. srcrsuggests = splitrsuggests
  354. srcrprovides = splitrprovides
  355. srcrreplaces = splitrreplaces
  356. srcrconflicts = splitrconflicts
  357. srcrpreinst = splitrpreinst
  358. srcrpostinst = splitrpostinst
  359. srcrprerm = splitrprerm
  360. srcrpostrm = splitrpostrm
  361. file_list = []
  362. walk_files(root, file_list, conffiles)
  363. if not file_list and localdata.getVar('ALLOW_EMPTY', False) != "1":
  364. bb.note("Not creating empty RPM package for %s" % splitname)
  365. else:
  366. spec_files_top.append('%files')
  367. if extra_pkgdata:
  368. package_rpm_extra_pkgdata(splitname, spec_files_top, localdata)
  369. spec_files_top.append('%defattr(-,-,-,-)')
  370. if file_list:
  371. bb.note("Creating RPM package for %s" % splitname)
  372. spec_files_top.extend(file_list)
  373. else:
  374. bb.note("Creating empty RPM package for %s" % splitname)
  375. spec_files_top.append('')
  376. continue
  377. # Process subpackage data
  378. spec_preamble_bottom.append('%%package -n %s' % splitname)
  379. spec_preamble_bottom.append('Summary: %s' % splitsummary)
  380. if srcversion != splitversion:
  381. spec_preamble_bottom.append('Version: %s' % splitversion)
  382. if srcrelease != splitrelease:
  383. spec_preamble_bottom.append('Release: %s' % splitrelease)
  384. if srcepoch != splitepoch:
  385. spec_preamble_bottom.append('Epoch: %s' % splitepoch)
  386. spec_preamble_bottom.append('License: %s' % splitlicense)
  387. spec_preamble_bottom.append('Group: %s' % splitsection)
  388. if srccustomtagschunk != splitcustomtagschunk:
  389. spec_preamble_bottom.append(splitcustomtagschunk)
  390. # Replaces == Obsoletes && Provides
  391. robsoletes = bb.utils.explode_dep_versions2(splitrobsoletes)
  392. rprovides = bb.utils.explode_dep_versions2(splitrprovides)
  393. rreplaces = bb.utils.explode_dep_versions2(splitrreplaces)
  394. for dep in rreplaces:
  395. if dep not in robsoletes:
  396. robsoletes[dep] = rreplaces[dep]
  397. if dep not in rprovides:
  398. rprovides[dep] = rreplaces[dep]
  399. splitrobsoletes = bb.utils.join_deps(robsoletes, commasep=False)
  400. splitrprovides = bb.utils.join_deps(rprovides, commasep=False)
  401. print_deps(splitrdepends, "Requires", spec_preamble_bottom, d)
  402. if splitrpreinst:
  403. print_deps(splitrdepends, "Requires(pre)", spec_preamble_bottom, d)
  404. if splitrpostinst:
  405. print_deps(splitrdepends, "Requires(post)", spec_preamble_bottom, d)
  406. if splitrprerm:
  407. print_deps(splitrdepends, "Requires(preun)", spec_preamble_bottom, d)
  408. if splitrpostrm:
  409. print_deps(splitrdepends, "Requires(postun)", spec_preamble_bottom, d)
  410. print_deps(splitrrecommends, "Recommends", spec_preamble_bottom, d)
  411. print_deps(splitrsuggests, "Suggests", spec_preamble_bottom, d)
  412. print_deps(splitrprovides, "Provides", spec_preamble_bottom, d)
  413. print_deps(splitrobsoletes, "Obsoletes", spec_preamble_bottom, d)
  414. print_deps(splitrconflicts, "Conflicts", spec_preamble_bottom, d)
  415. spec_preamble_bottom.append('')
  416. spec_preamble_bottom.append('%%description -n %s' % splitname)
  417. append_description(spec_preamble_bottom, splitdescription)
  418. spec_preamble_bottom.append('')
  419. # Now process scriptlets
  420. if splitrpreinst:
  421. spec_scriptlets_bottom.append('%%pre -n %s' % splitname)
  422. spec_scriptlets_bottom.append('# %s - preinst' % splitname)
  423. spec_scriptlets_bottom.append(splitrpreinst)
  424. spec_scriptlets_bottom.append('')
  425. if splitrpostinst:
  426. spec_scriptlets_bottom.append('%%post -n %s' % splitname)
  427. spec_scriptlets_bottom.append('# %s - postinst' % splitname)
  428. spec_scriptlets_bottom.append(splitrpostinst)
  429. spec_scriptlets_bottom.append('')
  430. if splitrprerm:
  431. spec_scriptlets_bottom.append('%%preun -n %s' % splitname)
  432. spec_scriptlets_bottom.append('# %s - prerm' % splitname)
  433. scriptvar = wrap_uninstall(splitrprerm)
  434. spec_scriptlets_bottom.append(scriptvar)
  435. spec_scriptlets_bottom.append('')
  436. if splitrpostrm:
  437. spec_scriptlets_bottom.append('%%postun -n %s' % splitname)
  438. spec_scriptlets_bottom.append('# %s - postrm' % splitname)
  439. scriptvar = wrap_uninstall(splitrpostrm)
  440. spec_scriptlets_bottom.append(scriptvar)
  441. spec_scriptlets_bottom.append('')
  442. # Now process files
  443. file_list = []
  444. walk_files(root, file_list, conffiles)
  445. if not file_list and localdata.getVar('ALLOW_EMPTY', False) != "1":
  446. bb.note("Not creating empty RPM package for %s" % splitname)
  447. else:
  448. spec_files_bottom.append('%%files -n %s' % splitname)
  449. if extra_pkgdata:
  450. package_rpm_extra_pkgdata(splitname, spec_files_bottom, localdata)
  451. spec_files_bottom.append('%defattr(-,-,-,-)')
  452. if file_list:
  453. bb.note("Creating RPM package for %s" % splitname)
  454. spec_files_bottom.extend(file_list)
  455. else:
  456. bb.note("Creating empty RPM package for %s" % splitname)
  457. spec_files_bottom.append('')
  458. del localdata
  459. add_prep(d, spec_files_bottom)
  460. spec_preamble_top.append('Summary: %s' % srcsummary)
  461. spec_preamble_top.append('Name: %s' % srcname)
  462. spec_preamble_top.append('Version: %s' % srcversion)
  463. spec_preamble_top.append('Release: %s' % srcrelease)
  464. if srcepoch and srcepoch.strip() != "":
  465. spec_preamble_top.append('Epoch: %s' % srcepoch)
  466. spec_preamble_top.append('License: %s' % srclicense)
  467. spec_preamble_top.append('Group: %s' % srcsection)
  468. spec_preamble_top.append('Packager: %s' % srcmaintainer)
  469. if srchomepage:
  470. spec_preamble_top.append('URL: %s' % srchomepage)
  471. if srccustomtagschunk:
  472. spec_preamble_top.append(srccustomtagschunk)
  473. tail_source(d)
  474. # Replaces == Obsoletes && Provides
  475. robsoletes = bb.utils.explode_dep_versions2(srcrobsoletes)
  476. rprovides = bb.utils.explode_dep_versions2(srcrprovides)
  477. rreplaces = bb.utils.explode_dep_versions2(srcrreplaces)
  478. for dep in rreplaces:
  479. if dep not in robsoletes:
  480. robsoletes[dep] = rreplaces[dep]
  481. if dep not in rprovides:
  482. rprovides[dep] = rreplaces[dep]
  483. srcrobsoletes = bb.utils.join_deps(robsoletes, commasep=False)
  484. srcrprovides = bb.utils.join_deps(rprovides, commasep=False)
  485. print_deps(srcdepends, "BuildRequires", spec_preamble_top, d)
  486. print_deps(srcrdepends, "Requires", spec_preamble_top, d)
  487. if srcrpreinst:
  488. print_deps(srcrdepends, "Requires(pre)", spec_preamble_top, d)
  489. if srcrpostinst:
  490. print_deps(srcrdepends, "Requires(post)", spec_preamble_top, d)
  491. if srcrprerm:
  492. print_deps(srcrdepends, "Requires(preun)", spec_preamble_top, d)
  493. if srcrpostrm:
  494. print_deps(srcrdepends, "Requires(postun)", spec_preamble_top, d)
  495. print_deps(srcrrecommends, "Recommends", spec_preamble_top, d)
  496. print_deps(srcrsuggests, "Suggests", spec_preamble_top, d)
  497. print_deps(srcrprovides, "Provides", spec_preamble_top, d)
  498. print_deps(srcrobsoletes, "Obsoletes", spec_preamble_top, d)
  499. print_deps(srcrconflicts, "Conflicts", spec_preamble_top, d)
  500. spec_preamble_top.append('')
  501. spec_preamble_top.append('%description')
  502. append_description(spec_preamble_top, srcdescription)
  503. spec_preamble_top.append('')
  504. if srcrpreinst:
  505. spec_scriptlets_top.append('%pre')
  506. spec_scriptlets_top.append('# %s - preinst' % srcname)
  507. spec_scriptlets_top.append(srcrpreinst)
  508. spec_scriptlets_top.append('')
  509. if srcrpostinst:
  510. spec_scriptlets_top.append('%post')
  511. spec_scriptlets_top.append('# %s - postinst' % srcname)
  512. spec_scriptlets_top.append(srcrpostinst)
  513. spec_scriptlets_top.append('')
  514. if srcrprerm:
  515. spec_scriptlets_top.append('%preun')
  516. spec_scriptlets_top.append('# %s - prerm' % srcname)
  517. scriptvar = wrap_uninstall(srcrprerm)
  518. spec_scriptlets_top.append(scriptvar)
  519. spec_scriptlets_top.append('')
  520. if srcrpostrm:
  521. spec_scriptlets_top.append('%postun')
  522. spec_scriptlets_top.append('# %s - postrm' % srcname)
  523. scriptvar = wrap_uninstall(srcrpostrm)
  524. spec_scriptlets_top.append(scriptvar)
  525. spec_scriptlets_top.append('')
  526. # Write the SPEC file
  527. specfile = open(outspecfile, 'w')
  528. # RPMSPEC_PREAMBLE is a way to add arbitrary text to the top
  529. # of the generated spec file
  530. external_preamble = d.getVar("RPMSPEC_PREAMBLE")
  531. if external_preamble:
  532. specfile.write(external_preamble + "\n")
  533. for line in spec_preamble_top:
  534. specfile.write(line + "\n")
  535. for line in spec_preamble_bottom:
  536. specfile.write(line + "\n")
  537. for line in spec_scriptlets_top:
  538. specfile.write(line + "\n")
  539. for line in spec_scriptlets_bottom:
  540. specfile.write(line + "\n")
  541. for line in spec_files_top:
  542. specfile.write(line + "\n")
  543. for line in spec_files_bottom:
  544. specfile.write(line + "\n")
  545. specfile.close()
  546. }
  547. # Otherwise allarch packages may change depending on override configuration
  548. write_specfile[vardepsexclude] = "OVERRIDES"
  549. # Have to list any variables referenced as X_<pkg> that aren't in pkgdata here
  550. RPMEXTRAVARS = "PACKAGE_ADD_METADATA_RPM"
  551. write_specfile[vardeps] += "${@gen_packagevar(d, 'RPMEXTRAVARS')}"
  552. python do_package_rpm () {
  553. workdir = d.getVar('WORKDIR')
  554. tmpdir = d.getVar('TMPDIR')
  555. pkgd = d.getVar('PKGD')
  556. if not workdir or not pkgd or not tmpdir:
  557. bb.error("Variables incorrectly set, unable to package")
  558. return
  559. packages = d.getVar('PACKAGES')
  560. if not packages or packages == '':
  561. bb.debug(1, "No packages; nothing to do")
  562. return
  563. # Construct the spec file...
  564. # If the spec file already exist, and has not been stored into
  565. # pseudo's files.db, it maybe cause rpmbuild src.rpm fail,
  566. # so remove it before doing rpmbuild src.rpm.
  567. srcname = d.getVar('PN')
  568. outspecfile = workdir + "/" + srcname + ".spec"
  569. if os.path.isfile(outspecfile):
  570. os.remove(outspecfile)
  571. d.setVar('OUTSPECFILE', outspecfile)
  572. bb.build.exec_func('write_specfile', d)
  573. perfiledeps = (d.getVar("MERGEPERFILEDEPS") or "0") == "0"
  574. if perfiledeps:
  575. outdepends, outprovides = write_rpm_perfiledata(srcname, d)
  576. # Setup the rpmbuild arguments...
  577. rpmbuild = d.getVar('RPMBUILD')
  578. rpmbuild_compmode = d.getVar('RPMBUILD_COMPMODE')
  579. rpmbuild_extra_params = d.getVar('RPMBUILD_EXTRA_PARAMS') or ""
  580. # Too many places in dnf stack assume that arch-independent packages are "noarch".
  581. # Let's not fight against this.
  582. package_arch = (d.getVar('PACKAGE_ARCH') or "").replace("-", "_")
  583. if package_arch == "all":
  584. package_arch = "noarch"
  585. d.setVar('PACKAGE_ARCH_EXTEND', package_arch)
  586. pkgwritedir = d.expand('${PKGWRITEDIRRPM}/${PACKAGE_ARCH_EXTEND}')
  587. d.setVar('RPM_PKGWRITEDIR', pkgwritedir)
  588. bb.debug(1, 'PKGWRITEDIR: %s' % d.getVar('RPM_PKGWRITEDIR'))
  589. pkgarch = d.expand('${PACKAGE_ARCH_EXTEND}${HOST_VENDOR}-linux')
  590. bb.utils.mkdirhier(pkgwritedir)
  591. os.chmod(pkgwritedir, 0o755)
  592. cmd = rpmbuild
  593. cmd = cmd + " --noclean --nodeps --short-circuit --target " + pkgarch + " --buildroot " + pkgd
  594. cmd = cmd + " --define '_topdir " + workdir + "' --define '_rpmdir " + pkgwritedir + "'"
  595. cmd = cmd + " --define '_builddir " + d.getVar('B') + "'"
  596. cmd = cmd + " --define '_build_name_fmt %%{NAME}-%%{VERSION}-%%{RELEASE}.%%{ARCH}.rpm'"
  597. cmd = cmd + " --define '_use_internal_dependency_generator 0'"
  598. cmd = cmd + " --define '_binaries_in_noarch_packages_terminate_build 0'"
  599. cmd = cmd + " --define '_build_id_links none'"
  600. cmd = cmd + " --define '_smp_ncpus_max 4'"
  601. cmd = cmd + " --define '_source_payload %s'" % rpmbuild_compmode
  602. cmd = cmd + " --define '_binary_payload %s'" % rpmbuild_compmode
  603. cmd = cmd + " --define 'build_mtime_policy clamp_to_source_date_epoch'"
  604. cmd = cmd + " --define 'use_source_date_epoch_as_buildtime 1'"
  605. cmd = cmd + " --define '_buildhost reproducible'"
  606. cmd = cmd + " --define '__font_provides %{nil}'"
  607. if perfiledeps:
  608. cmd = cmd + " --define '__find_requires " + outdepends + "'"
  609. cmd = cmd + " --define '__find_provides " + outprovides + "'"
  610. else:
  611. cmd = cmd + " --define '__find_requires %{nil}'"
  612. cmd = cmd + " --define '__find_provides %{nil}'"
  613. cmd = cmd + " --define '_unpackaged_files_terminate_build 0'"
  614. cmd = cmd + " --define 'debug_package %{nil}'"
  615. cmd = cmd + " --define '_tmppath " + workdir + "'"
  616. cmd = cmd + " --define '_use_weak_usergroup_deps 1'"
  617. cmd = cmd + " --define '_passwd_path " + "/completely/bogus/path" + "'"
  618. cmd = cmd + " --define '_group_path " + "/completely/bogus/path" + "'"
  619. cmd = cmd + rpmbuild_extra_params
  620. if d.getVarFlag('ARCHIVER_MODE', 'srpm') == '1' and bb.data.inherits_class('archiver', d):
  621. cmd = cmd + " --define '_sourcedir " + d.getVar('ARCHIVER_OUTDIR') + "'"
  622. cmdsrpm = cmd + " --define '_srcrpmdir " + d.getVar('ARCHIVER_RPMOUTDIR') + "'"
  623. cmdsrpm = cmdsrpm + " -bs " + outspecfile
  624. # Build the .src.rpm
  625. d.setVar('SBUILDSPEC', cmdsrpm + "\n")
  626. d.setVarFlag('SBUILDSPEC', 'func', '1')
  627. bb.build.exec_func('SBUILDSPEC', d)
  628. cmd = cmd + " -bb " + outspecfile
  629. # rpm 4 creates various empty directories in _topdir, let's clean them up
  630. cleanupcmd = "rm -rf %s/BUILDROOT %s/SOURCES %s/SPECS %s/SRPMS" % (workdir, workdir, workdir, workdir)
  631. # Build the rpm package!
  632. d.setVar('BUILDSPEC', cmd + "\n" + cleanupcmd + "\n")
  633. d.setVarFlag('BUILDSPEC', 'func', '1')
  634. bb.build.exec_func('BUILDSPEC', d)
  635. if d.getVar('RPM_SIGN_PACKAGES') == '1':
  636. bb.build.exec_func("sign_rpm", d)
  637. }
  638. python () {
  639. if d.getVar('PACKAGES') != '':
  640. deps = ' rpm-native:do_populate_sysroot virtual/fakeroot-native:do_populate_sysroot'
  641. d.appendVarFlag('do_package_write_rpm', 'depends', deps)
  642. d.setVarFlag('do_package_write_rpm', 'fakeroot', '1')
  643. # Needed to ensure PKG_xxx renaming of dependency packages works
  644. d.setVarFlag('do_package_write_rpm', 'deptask', "do_packagedata")
  645. d.setVarFlag('do_package_write_rpm', 'rdeptask', "do_packagedata")
  646. }
  647. SSTATETASKS += "do_package_write_rpm"
  648. do_package_write_rpm[sstate-inputdirs] = "${PKGWRITEDIRRPM}"
  649. do_package_write_rpm[sstate-outputdirs] = "${DEPLOY_DIR_RPM}"
  650. # Take a shared lock, we can write multiple packages at the same time...
  651. # but we need to stop the rootfs/solver from running while we do...
  652. do_package_write_rpm[sstate-lockfile-shared] += "${DEPLOY_DIR_RPM}/rpm.lock"
  653. python do_package_write_rpm_setscene () {
  654. sstate_setscene(d)
  655. }
  656. addtask do_package_write_rpm_setscene
  657. python do_package_write_rpm () {
  658. bb.build.exec_func("read_subpackage_metadata", d)
  659. bb.build.exec_func("do_package_rpm", d)
  660. }
  661. do_package_write_rpm[dirs] = "${PKGWRITEDIRRPM}"
  662. do_package_write_rpm[cleandirs] = "${PKGWRITEDIRRPM}"
  663. do_package_write_rpm[depends] += "${@oe.utils.build_depends_string(d.getVar('PACKAGE_WRITE_DEPS'), 'do_populate_sysroot')}"
  664. addtask package_write_rpm after do_packagedata do_package do_deploy_source_date_epoch before do_build
  665. do_build[rdeptask] += "do_package_write_rpm"
  666. PACKAGEINDEXDEPS += "rpm-native:do_populate_sysroot"
  667. PACKAGEINDEXDEPS += "createrepo-c-native:do_populate_sysroot"