894 lines
39 KiB
Python
894 lines
39 KiB
Python
# Recipe creation tool - create command build system handlers
|
|
#
|
|
# Copyright (C) 2014-2016 Intel Corporation
|
|
#
|
|
# This program is free software; you can redistribute it and/or modify
|
|
# it under the terms of the GNU General Public License version 2 as
|
|
# published by the Free Software Foundation.
|
|
#
|
|
# This program is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
# GNU General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU General Public License along
|
|
# with this program; if not, write to the Free Software Foundation, Inc.,
|
|
# 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
|
|
import re
|
|
import logging
|
|
import glob
|
|
from recipetool.create import RecipeHandler, validate_pv
|
|
|
|
logger = logging.getLogger('recipetool')
|
|
|
|
tinfoil = None
|
|
plugins = None
|
|
|
|
def plugin_init(pluginlist):
|
|
# Take a reference to the list so we can use it later
|
|
global plugins
|
|
plugins = pluginlist
|
|
|
|
def tinfoil_init(instance):
|
|
global tinfoil
|
|
tinfoil = instance
|
|
|
|
|
|
class CmakeRecipeHandler(RecipeHandler):
|
|
def process(self, srctree, classes, lines_before, lines_after, handled, extravalues):
|
|
if 'buildsystem' in handled:
|
|
return False
|
|
|
|
if RecipeHandler.checkfiles(srctree, ['CMakeLists.txt']):
|
|
classes.append('cmake')
|
|
values = CmakeRecipeHandler.extract_cmake_deps(lines_before, srctree, extravalues)
|
|
classes.extend(values.pop('inherit', '').split())
|
|
for var, value in values.items():
|
|
lines_before.append('%s = "%s"' % (var, value))
|
|
lines_after.append('# Specify any options you want to pass to cmake using EXTRA_OECMAKE:')
|
|
lines_after.append('EXTRA_OECMAKE = ""')
|
|
lines_after.append('')
|
|
handled.append('buildsystem')
|
|
return True
|
|
return False
|
|
|
|
@staticmethod
|
|
def extract_cmake_deps(outlines, srctree, extravalues, cmakelistsfile=None):
|
|
# Find all plugins that want to register handlers
|
|
logger.debug('Loading cmake handlers')
|
|
handlers = []
|
|
for plugin in plugins:
|
|
if hasattr(plugin, 'register_cmake_handlers'):
|
|
plugin.register_cmake_handlers(handlers)
|
|
|
|
values = {}
|
|
inherits = []
|
|
|
|
if cmakelistsfile:
|
|
srcfiles = [cmakelistsfile]
|
|
else:
|
|
srcfiles = RecipeHandler.checkfiles(srctree, ['CMakeLists.txt'])
|
|
|
|
# Note that some of these are non-standard, but probably better to
|
|
# be able to map them anyway if we see them
|
|
cmake_pkgmap = {'alsa': 'alsa-lib',
|
|
'aspell': 'aspell',
|
|
'atk': 'atk',
|
|
'bison': 'bison-native',
|
|
'boost': 'boost',
|
|
'bzip2': 'bzip2',
|
|
'cairo': 'cairo',
|
|
'cups': 'cups',
|
|
'curl': 'curl',
|
|
'curses': 'ncurses',
|
|
'cvs': 'cvs',
|
|
'drm': 'libdrm',
|
|
'dbus': 'dbus',
|
|
'dbusglib': 'dbus-glib',
|
|
'egl': 'virtual/egl',
|
|
'expat': 'expat',
|
|
'flex': 'flex-native',
|
|
'fontconfig': 'fontconfig',
|
|
'freetype': 'freetype',
|
|
'gettext': '',
|
|
'git': '',
|
|
'gio': 'glib-2.0',
|
|
'giounix': 'glib-2.0',
|
|
'glew': 'glew',
|
|
'glib': 'glib-2.0',
|
|
'glib2': 'glib-2.0',
|
|
'glu': 'libglu',
|
|
'glut': 'freeglut',
|
|
'gobject': 'glib-2.0',
|
|
'gperf': 'gperf-native',
|
|
'gnutls': 'gnutls',
|
|
'gtk2': 'gtk+',
|
|
'gtk3': 'gtk+3',
|
|
'gtk': 'gtk+3',
|
|
'harfbuzz': 'harfbuzz',
|
|
'icu': 'icu',
|
|
'intl': 'virtual/libintl',
|
|
'jpeg': 'jpeg',
|
|
'libarchive': 'libarchive',
|
|
'libiconv': 'virtual/libiconv',
|
|
'liblzma': 'xz',
|
|
'libxml2': 'libxml2',
|
|
'libxslt': 'libxslt',
|
|
'opengl': 'virtual/libgl',
|
|
'openmp': '',
|
|
'openssl': 'openssl',
|
|
'pango': 'pango',
|
|
'perl': '',
|
|
'perllibs': '',
|
|
'pkgconfig': '',
|
|
'png': 'libpng',
|
|
'pthread': '',
|
|
'pythoninterp': '',
|
|
'pythonlibs': '',
|
|
'ruby': 'ruby-native',
|
|
'sdl': 'libsdl',
|
|
'sdl2': 'libsdl2',
|
|
'subversion': 'subversion-native',
|
|
'swig': 'swig-native',
|
|
'tcl': 'tcl-native',
|
|
'threads': '',
|
|
'tiff': 'tiff',
|
|
'wget': 'wget',
|
|
'x11': 'libx11',
|
|
'xcb': 'libxcb',
|
|
'xext': 'libxext',
|
|
'xfixes': 'libxfixes',
|
|
'zlib': 'zlib',
|
|
}
|
|
|
|
pcdeps = []
|
|
libdeps = []
|
|
deps = []
|
|
unmappedpkgs = []
|
|
|
|
proj_re = re.compile('project\s*\(([^)]*)\)', re.IGNORECASE)
|
|
pkgcm_re = re.compile('pkg_check_modules\s*\(\s*[a-zA-Z0-9-_]+\s*(REQUIRED)?\s+([^)\s]+)\s*\)', re.IGNORECASE)
|
|
pkgsm_re = re.compile('pkg_search_module\s*\(\s*[a-zA-Z0-9-_]+\s*(REQUIRED)?((\s+[^)\s]+)+)\s*\)', re.IGNORECASE)
|
|
findpackage_re = re.compile('find_package\s*\(\s*([a-zA-Z0-9-_]+)\s*.*', re.IGNORECASE)
|
|
findlibrary_re = re.compile('find_library\s*\(\s*[a-zA-Z0-9-_]+\s*(NAMES\s+)?([a-zA-Z0-9-_ ]+)\s*.*')
|
|
checklib_re = re.compile('check_library_exists\s*\(\s*([^\s)]+)\s*.*', re.IGNORECASE)
|
|
include_re = re.compile('include\s*\(\s*([^)\s]*)\s*\)', re.IGNORECASE)
|
|
subdir_re = re.compile('add_subdirectory\s*\(\s*([^)\s]*)\s*([^)\s]*)\s*\)', re.IGNORECASE)
|
|
dep_re = re.compile('([^ ><=]+)( *[<>=]+ *[^ ><=]+)?')
|
|
|
|
def find_cmake_package(pkg):
|
|
RecipeHandler.load_devel_filemap(tinfoil.config_data)
|
|
for fn, pn in RecipeHandler.recipecmakefilemap.items():
|
|
splitname = fn.split('/')
|
|
if len(splitname) > 1:
|
|
if splitname[0].lower().startswith(pkg.lower()):
|
|
if splitname[1] == '%s-config.cmake' % pkg.lower() or splitname[1] == '%sConfig.cmake' % pkg or splitname[1] == 'Find%s.cmake' % pkg:
|
|
return pn
|
|
return None
|
|
|
|
def interpret_value(value):
|
|
return value.strip('"')
|
|
|
|
def parse_cmake_file(fn, paths=None):
|
|
searchpaths = (paths or []) + [os.path.dirname(fn)]
|
|
logger.debug('Parsing file %s' % fn)
|
|
with open(fn, 'r', errors='surrogateescape') as f:
|
|
for line in f:
|
|
line = line.strip()
|
|
for handler in handlers:
|
|
if handler.process_line(srctree, fn, line, libdeps, pcdeps, deps, outlines, inherits, values):
|
|
continue
|
|
res = include_re.match(line)
|
|
if res:
|
|
includefn = bb.utils.which(':'.join(searchpaths), res.group(1))
|
|
if includefn:
|
|
parse_cmake_file(includefn, searchpaths)
|
|
else:
|
|
logger.debug('Unable to recurse into include file %s' % res.group(1))
|
|
continue
|
|
res = subdir_re.match(line)
|
|
if res:
|
|
subdirfn = os.path.join(os.path.dirname(fn), res.group(1), 'CMakeLists.txt')
|
|
if os.path.exists(subdirfn):
|
|
parse_cmake_file(subdirfn, searchpaths)
|
|
else:
|
|
logger.debug('Unable to recurse into subdirectory file %s' % subdirfn)
|
|
continue
|
|
res = proj_re.match(line)
|
|
if res:
|
|
extravalues['PN'] = interpret_value(res.group(1).split()[0])
|
|
continue
|
|
res = pkgcm_re.match(line)
|
|
if res:
|
|
res = dep_re.findall(res.group(2))
|
|
if res:
|
|
pcdeps.extend([interpret_value(x[0]) for x in res])
|
|
inherits.append('pkgconfig')
|
|
continue
|
|
res = pkgsm_re.match(line)
|
|
if res:
|
|
res = dep_re.findall(res.group(2))
|
|
if res:
|
|
# Note: appending a tuple here!
|
|
item = tuple((interpret_value(x[0]) for x in res))
|
|
if len(item) == 1:
|
|
item = item[0]
|
|
pcdeps.append(item)
|
|
inherits.append('pkgconfig')
|
|
continue
|
|
res = findpackage_re.match(line)
|
|
if res:
|
|
origpkg = res.group(1)
|
|
pkg = interpret_value(origpkg)
|
|
found = False
|
|
for handler in handlers:
|
|
if handler.process_findpackage(srctree, fn, pkg, deps, outlines, inherits, values):
|
|
logger.debug('Mapped CMake package %s via handler %s' % (pkg, handler.__class__.__name__))
|
|
found = True
|
|
break
|
|
if found:
|
|
continue
|
|
elif pkg == 'Gettext':
|
|
inherits.append('gettext')
|
|
elif pkg == 'Perl':
|
|
inherits.append('perlnative')
|
|
elif pkg == 'PkgConfig':
|
|
inherits.append('pkgconfig')
|
|
elif pkg == 'PythonInterp':
|
|
inherits.append('pythonnative')
|
|
elif pkg == 'PythonLibs':
|
|
inherits.append('python-dir')
|
|
else:
|
|
# Try to map via looking at installed CMake packages in pkgdata
|
|
dep = find_cmake_package(pkg)
|
|
if dep:
|
|
logger.debug('Mapped CMake package %s to recipe %s via pkgdata' % (pkg, dep))
|
|
deps.append(dep)
|
|
else:
|
|
dep = cmake_pkgmap.get(pkg.lower(), None)
|
|
if dep:
|
|
logger.debug('Mapped CMake package %s to recipe %s via internal list' % (pkg, dep))
|
|
deps.append(dep)
|
|
elif dep is None:
|
|
unmappedpkgs.append(origpkg)
|
|
continue
|
|
res = checklib_re.match(line)
|
|
if res:
|
|
lib = interpret_value(res.group(1))
|
|
if not lib.startswith('$'):
|
|
libdeps.append(lib)
|
|
res = findlibrary_re.match(line)
|
|
if res:
|
|
libs = res.group(2).split()
|
|
for lib in libs:
|
|
if lib in ['HINTS', 'PATHS', 'PATH_SUFFIXES', 'DOC', 'NAMES_PER_DIR'] or lib.startswith(('NO_', 'CMAKE_', 'ONLY_CMAKE_')):
|
|
break
|
|
lib = interpret_value(lib)
|
|
if not lib.startswith('$'):
|
|
libdeps.append(lib)
|
|
if line.lower().startswith('useswig'):
|
|
deps.append('swig-native')
|
|
continue
|
|
|
|
parse_cmake_file(srcfiles[0])
|
|
|
|
if unmappedpkgs:
|
|
outlines.append('# NOTE: unable to map the following CMake package dependencies: %s' % ' '.join(list(set(unmappedpkgs))))
|
|
|
|
RecipeHandler.handle_depends(libdeps, pcdeps, deps, outlines, values, tinfoil.config_data)
|
|
|
|
for handler in handlers:
|
|
handler.post_process(srctree, libdeps, pcdeps, deps, outlines, inherits, values)
|
|
|
|
if inherits:
|
|
values['inherit'] = ' '.join(list(set(inherits)))
|
|
|
|
return values
|
|
|
|
|
|
class CmakeExtensionHandler(object):
|
|
'''Base class for CMake extension handlers'''
|
|
def process_line(self, srctree, fn, line, libdeps, pcdeps, deps, outlines, inherits, values):
|
|
'''
|
|
Handle a line parsed out of an CMake file.
|
|
Return True if you've completely handled the passed in line, otherwise return False.
|
|
'''
|
|
return False
|
|
|
|
def process_findpackage(self, srctree, fn, pkg, deps, outlines, inherits, values):
|
|
'''
|
|
Handle a find_package package parsed out of a CMake file.
|
|
Return True if you've completely handled the passed in package, otherwise return False.
|
|
'''
|
|
return False
|
|
|
|
def post_process(self, srctree, fn, pkg, deps, outlines, inherits, values):
|
|
'''
|
|
Apply any desired post-processing on the output
|
|
'''
|
|
return
|
|
|
|
|
|
|
|
class SconsRecipeHandler(RecipeHandler):
|
|
def process(self, srctree, classes, lines_before, lines_after, handled, extravalues):
|
|
if 'buildsystem' in handled:
|
|
return False
|
|
|
|
if RecipeHandler.checkfiles(srctree, ['SConstruct', 'Sconstruct', 'sconstruct']):
|
|
classes.append('scons')
|
|
lines_after.append('# Specify any options you want to pass to scons using EXTRA_OESCONS:')
|
|
lines_after.append('EXTRA_OESCONS = ""')
|
|
lines_after.append('')
|
|
handled.append('buildsystem')
|
|
return True
|
|
return False
|
|
|
|
|
|
class QmakeRecipeHandler(RecipeHandler):
|
|
def process(self, srctree, classes, lines_before, lines_after, handled, extravalues):
|
|
if 'buildsystem' in handled:
|
|
return False
|
|
|
|
if RecipeHandler.checkfiles(srctree, ['*.pro']):
|
|
classes.append('qmake2')
|
|
handled.append('buildsystem')
|
|
return True
|
|
return False
|
|
|
|
|
|
class AutotoolsRecipeHandler(RecipeHandler):
|
|
def process(self, srctree, classes, lines_before, lines_after, handled, extravalues):
|
|
if 'buildsystem' in handled:
|
|
return False
|
|
|
|
autoconf = False
|
|
if RecipeHandler.checkfiles(srctree, ['configure.ac', 'configure.in']):
|
|
autoconf = True
|
|
values = AutotoolsRecipeHandler.extract_autotools_deps(lines_before, srctree, extravalues)
|
|
classes.extend(values.pop('inherit', '').split())
|
|
for var, value in values.items():
|
|
lines_before.append('%s = "%s"' % (var, value))
|
|
else:
|
|
conffile = RecipeHandler.checkfiles(srctree, ['configure'])
|
|
if conffile:
|
|
# Check if this is just a pre-generated autoconf configure script
|
|
with open(conffile[0], 'r', errors='surrogateescape') as f:
|
|
for i in range(1, 10):
|
|
if 'Generated by GNU Autoconf' in f.readline():
|
|
autoconf = True
|
|
break
|
|
|
|
if autoconf and not ('PV' in extravalues and 'PN' in extravalues):
|
|
# Last resort
|
|
conffile = RecipeHandler.checkfiles(srctree, ['configure'])
|
|
if conffile:
|
|
with open(conffile[0], 'r', errors='surrogateescape') as f:
|
|
for line in f:
|
|
line = line.strip()
|
|
if line.startswith('VERSION=') or line.startswith('PACKAGE_VERSION='):
|
|
pv = line.split('=')[1].strip('"\'')
|
|
if pv and not 'PV' in extravalues and validate_pv(pv):
|
|
extravalues['PV'] = pv
|
|
elif line.startswith('PACKAGE_NAME=') or line.startswith('PACKAGE='):
|
|
pn = line.split('=')[1].strip('"\'')
|
|
if pn and not 'PN' in extravalues:
|
|
extravalues['PN'] = pn
|
|
|
|
if autoconf:
|
|
lines_before.append('')
|
|
lines_before.append('# NOTE: if this software is not capable of being built in a separate build directory')
|
|
lines_before.append('# from the source, you should replace autotools with autotools-brokensep in the')
|
|
lines_before.append('# inherit line')
|
|
classes.append('autotools')
|
|
lines_after.append('# Specify any options you want to pass to the configure script using EXTRA_OECONF:')
|
|
lines_after.append('EXTRA_OECONF = ""')
|
|
lines_after.append('')
|
|
handled.append('buildsystem')
|
|
return True
|
|
|
|
return False
|
|
|
|
@staticmethod
|
|
def extract_autotools_deps(outlines, srctree, extravalues=None, acfile=None):
|
|
import shlex
|
|
|
|
# Find all plugins that want to register handlers
|
|
logger.debug('Loading autotools handlers')
|
|
handlers = []
|
|
for plugin in plugins:
|
|
if hasattr(plugin, 'register_autotools_handlers'):
|
|
plugin.register_autotools_handlers(handlers)
|
|
|
|
values = {}
|
|
inherits = []
|
|
|
|
# Hardcoded map, we also use a dynamic one based on what's in the sysroot
|
|
progmap = {'flex': 'flex-native',
|
|
'bison': 'bison-native',
|
|
'm4': 'm4-native',
|
|
'tar': 'tar-native',
|
|
'ar': 'binutils-native',
|
|
'ranlib': 'binutils-native',
|
|
'ld': 'binutils-native',
|
|
'strip': 'binutils-native',
|
|
'libtool': '',
|
|
'autoconf': '',
|
|
'autoheader': '',
|
|
'automake': '',
|
|
'uname': '',
|
|
'rm': '',
|
|
'cp': '',
|
|
'mv': '',
|
|
'find': '',
|
|
'awk': '',
|
|
'sed': '',
|
|
}
|
|
progclassmap = {'gconftool-2': 'gconf',
|
|
'pkg-config': 'pkgconfig',
|
|
'python': 'pythonnative',
|
|
'python3': 'python3native',
|
|
'perl': 'perlnative',
|
|
'makeinfo': 'texinfo',
|
|
}
|
|
|
|
pkg_re = re.compile('PKG_CHECK_MODULES\(\s*\[?[a-zA-Z0-9_]*\]?,\s*\[?([^,\]]*)\]?[),].*')
|
|
pkgce_re = re.compile('PKG_CHECK_EXISTS\(\s*\[?([^,\]]*)\]?[),].*')
|
|
lib_re = re.compile('AC_CHECK_LIB\(\s*\[?([^,\]]*)\]?,.*')
|
|
libx_re = re.compile('AX_CHECK_LIBRARY\(\s*\[?[^,\]]*\]?,\s*\[?([^,\]]*)\]?,\s*\[?([a-zA-Z0-9-]*)\]?,.*')
|
|
progs_re = re.compile('_PROGS?\(\s*\[?[a-zA-Z0-9_]*\]?,\s*\[?([^,\]]*)\]?[),].*')
|
|
dep_re = re.compile('([^ ><=]+)( [<>=]+ [^ ><=]+)?')
|
|
ac_init_re = re.compile('AC_INIT\(\s*([^,]+),\s*([^,]+)[,)].*')
|
|
am_init_re = re.compile('AM_INIT_AUTOMAKE\(\s*([^,]+),\s*([^,]+)[,)].*')
|
|
define_re = re.compile('\s*(m4_)?define\(\s*([^,]+),\s*([^,]+)\)')
|
|
version_re = re.compile('([0-9.]+)')
|
|
|
|
defines = {}
|
|
def subst_defines(value):
|
|
newvalue = value
|
|
for define, defval in defines.items():
|
|
newvalue = newvalue.replace(define, defval)
|
|
if newvalue != value:
|
|
return subst_defines(newvalue)
|
|
return value
|
|
|
|
def process_value(value):
|
|
value = value.replace('[', '').replace(']', '')
|
|
if value.startswith('m4_esyscmd(') or value.startswith('m4_esyscmd_s('):
|
|
cmd = subst_defines(value[value.index('(')+1:-1])
|
|
try:
|
|
if '|' in cmd:
|
|
cmd = 'set -o pipefail; ' + cmd
|
|
stdout, _ = bb.process.run(cmd, cwd=srctree, shell=True)
|
|
ret = stdout.rstrip()
|
|
except bb.process.ExecutionError as e:
|
|
ret = ''
|
|
elif value.startswith('m4_'):
|
|
return None
|
|
ret = subst_defines(value)
|
|
if ret:
|
|
ret = ret.strip('"\'')
|
|
return ret
|
|
|
|
# Since a configure.ac file is essentially a program, this is only ever going to be
|
|
# a hack unfortunately; but it ought to be enough of an approximation
|
|
if acfile:
|
|
srcfiles = [acfile]
|
|
else:
|
|
srcfiles = RecipeHandler.checkfiles(srctree, ['acinclude.m4', 'configure.ac', 'configure.in'])
|
|
|
|
pcdeps = []
|
|
libdeps = []
|
|
deps = []
|
|
unmapped = []
|
|
|
|
RecipeHandler.load_binmap(tinfoil.config_data)
|
|
|
|
def process_macro(keyword, value):
|
|
for handler in handlers:
|
|
if handler.process_macro(srctree, keyword, value, process_value, libdeps, pcdeps, deps, outlines, inherits, values):
|
|
return
|
|
logger.debug('Found keyword %s with value "%s"' % (keyword, value))
|
|
if keyword == 'PKG_CHECK_MODULES':
|
|
res = pkg_re.search(value)
|
|
if res:
|
|
res = dep_re.findall(res.group(1))
|
|
if res:
|
|
pcdeps.extend([x[0] for x in res])
|
|
inherits.append('pkgconfig')
|
|
elif keyword == 'PKG_CHECK_EXISTS':
|
|
res = pkgce_re.search(value)
|
|
if res:
|
|
res = dep_re.findall(res.group(1))
|
|
if res:
|
|
pcdeps.extend([x[0] for x in res])
|
|
inherits.append('pkgconfig')
|
|
elif keyword in ('AM_GNU_GETTEXT', 'AM_GLIB_GNU_GETTEXT', 'GETTEXT_PACKAGE'):
|
|
inherits.append('gettext')
|
|
elif keyword in ('AC_PROG_INTLTOOL', 'IT_PROG_INTLTOOL'):
|
|
deps.append('intltool-native')
|
|
elif keyword == 'AM_PATH_GLIB_2_0':
|
|
deps.append('glib-2.0')
|
|
elif keyword in ('AC_CHECK_PROG', 'AC_PATH_PROG', 'AX_WITH_PROG'):
|
|
res = progs_re.search(value)
|
|
if res:
|
|
for prog in shlex.split(res.group(1)):
|
|
prog = prog.split()[0]
|
|
for handler in handlers:
|
|
if handler.process_prog(srctree, keyword, value, prog, deps, outlines, inherits, values):
|
|
return
|
|
progclass = progclassmap.get(prog, None)
|
|
if progclass:
|
|
inherits.append(progclass)
|
|
else:
|
|
progdep = RecipeHandler.recipebinmap.get(prog, None)
|
|
if not progdep:
|
|
progdep = progmap.get(prog, None)
|
|
if progdep:
|
|
deps.append(progdep)
|
|
elif progdep is None:
|
|
if not prog.startswith('$'):
|
|
unmapped.append(prog)
|
|
elif keyword == 'AC_CHECK_LIB':
|
|
res = lib_re.search(value)
|
|
if res:
|
|
lib = res.group(1)
|
|
if not lib.startswith('$'):
|
|
libdeps.append(lib)
|
|
elif keyword == 'AX_CHECK_LIBRARY':
|
|
res = libx_re.search(value)
|
|
if res:
|
|
lib = res.group(2)
|
|
if not lib.startswith('$'):
|
|
header = res.group(1)
|
|
libdeps.append((lib, header))
|
|
elif keyword == 'AC_PATH_X':
|
|
deps.append('libx11')
|
|
elif keyword in ('AX_BOOST', 'BOOST_REQUIRE'):
|
|
deps.append('boost')
|
|
elif keyword in ('AC_PROG_LEX', 'AM_PROG_LEX', 'AX_PROG_FLEX'):
|
|
deps.append('flex-native')
|
|
elif keyword in ('AC_PROG_YACC', 'AX_PROG_BISON'):
|
|
deps.append('bison-native')
|
|
elif keyword == 'AX_CHECK_ZLIB':
|
|
deps.append('zlib')
|
|
elif keyword in ('AX_CHECK_OPENSSL', 'AX_LIB_CRYPTO'):
|
|
deps.append('openssl')
|
|
elif keyword == 'AX_LIB_CURL':
|
|
deps.append('curl')
|
|
elif keyword == 'AX_LIB_BEECRYPT':
|
|
deps.append('beecrypt')
|
|
elif keyword == 'AX_LIB_EXPAT':
|
|
deps.append('expat')
|
|
elif keyword == 'AX_LIB_GCRYPT':
|
|
deps.append('libgcrypt')
|
|
elif keyword == 'AX_LIB_NETTLE':
|
|
deps.append('nettle')
|
|
elif keyword == 'AX_LIB_READLINE':
|
|
deps.append('readline')
|
|
elif keyword == 'AX_LIB_SQLITE3':
|
|
deps.append('sqlite3')
|
|
elif keyword == 'AX_LIB_TAGLIB':
|
|
deps.append('taglib')
|
|
elif keyword in ['AX_PKG_SWIG', 'AC_PROG_SWIG']:
|
|
deps.append('swig-native')
|
|
elif keyword == 'AX_PROG_XSLTPROC':
|
|
deps.append('libxslt-native')
|
|
elif keyword in ['AC_PYTHON_DEVEL', 'AX_PYTHON_DEVEL', 'AM_PATH_PYTHON']:
|
|
pythonclass = 'pythonnative'
|
|
res = version_re.search(value)
|
|
if res:
|
|
if res.group(1).startswith('3'):
|
|
pythonclass = 'python3native'
|
|
# Avoid replacing python3native with pythonnative
|
|
if not pythonclass in inherits and not 'python3native' in inherits:
|
|
if 'pythonnative' in inherits:
|
|
inherits.remove('pythonnative')
|
|
inherits.append(pythonclass)
|
|
elif keyword == 'AX_WITH_CURSES':
|
|
deps.append('ncurses')
|
|
elif keyword == 'AX_PATH_BDB':
|
|
deps.append('db')
|
|
elif keyword == 'AX_PATH_LIB_PCRE':
|
|
deps.append('libpcre')
|
|
elif keyword == 'AC_INIT':
|
|
if extravalues is not None:
|
|
res = ac_init_re.match(value)
|
|
if res:
|
|
extravalues['PN'] = process_value(res.group(1))
|
|
pv = process_value(res.group(2))
|
|
if validate_pv(pv):
|
|
extravalues['PV'] = pv
|
|
elif keyword == 'AM_INIT_AUTOMAKE':
|
|
if extravalues is not None:
|
|
if 'PN' not in extravalues:
|
|
res = am_init_re.match(value)
|
|
if res:
|
|
if res.group(1) != 'AC_PACKAGE_NAME':
|
|
extravalues['PN'] = process_value(res.group(1))
|
|
pv = process_value(res.group(2))
|
|
if validate_pv(pv):
|
|
extravalues['PV'] = pv
|
|
elif keyword == 'define(':
|
|
res = define_re.match(value)
|
|
if res:
|
|
key = res.group(2).strip('[]')
|
|
value = process_value(res.group(3))
|
|
if value is not None:
|
|
defines[key] = value
|
|
|
|
keywords = ['PKG_CHECK_MODULES',
|
|
'PKG_CHECK_EXISTS',
|
|
'AM_GNU_GETTEXT',
|
|
'AM_GLIB_GNU_GETTEXT',
|
|
'GETTEXT_PACKAGE',
|
|
'AC_PROG_INTLTOOL',
|
|
'IT_PROG_INTLTOOL',
|
|
'AM_PATH_GLIB_2_0',
|
|
'AC_CHECK_PROG',
|
|
'AC_PATH_PROG',
|
|
'AX_WITH_PROG',
|
|
'AC_CHECK_LIB',
|
|
'AX_CHECK_LIBRARY',
|
|
'AC_PATH_X',
|
|
'AX_BOOST',
|
|
'BOOST_REQUIRE',
|
|
'AC_PROG_LEX',
|
|
'AM_PROG_LEX',
|
|
'AX_PROG_FLEX',
|
|
'AC_PROG_YACC',
|
|
'AX_PROG_BISON',
|
|
'AX_CHECK_ZLIB',
|
|
'AX_CHECK_OPENSSL',
|
|
'AX_LIB_CRYPTO',
|
|
'AX_LIB_CURL',
|
|
'AX_LIB_BEECRYPT',
|
|
'AX_LIB_EXPAT',
|
|
'AX_LIB_GCRYPT',
|
|
'AX_LIB_NETTLE',
|
|
'AX_LIB_READLINE'
|
|
'AX_LIB_SQLITE3',
|
|
'AX_LIB_TAGLIB',
|
|
'AX_PKG_SWIG',
|
|
'AC_PROG_SWIG',
|
|
'AX_PROG_XSLTPROC',
|
|
'AC_PYTHON_DEVEL',
|
|
'AX_PYTHON_DEVEL',
|
|
'AM_PATH_PYTHON',
|
|
'AX_WITH_CURSES',
|
|
'AX_PATH_BDB',
|
|
'AX_PATH_LIB_PCRE',
|
|
'AC_INIT',
|
|
'AM_INIT_AUTOMAKE',
|
|
'define(',
|
|
]
|
|
|
|
for handler in handlers:
|
|
handler.extend_keywords(keywords)
|
|
|
|
for srcfile in srcfiles:
|
|
nesting = 0
|
|
in_keyword = ''
|
|
partial = ''
|
|
with open(srcfile, 'r', errors='surrogateescape') as f:
|
|
for line in f:
|
|
if in_keyword:
|
|
partial += ' ' + line.strip()
|
|
if partial.endswith('\\'):
|
|
partial = partial[:-1]
|
|
nesting = nesting + line.count('(') - line.count(')')
|
|
if nesting == 0:
|
|
process_macro(in_keyword, partial)
|
|
partial = ''
|
|
in_keyword = ''
|
|
else:
|
|
for keyword in keywords:
|
|
if keyword in line:
|
|
nesting = line.count('(') - line.count(')')
|
|
if nesting > 0:
|
|
partial = line.strip()
|
|
if partial.endswith('\\'):
|
|
partial = partial[:-1]
|
|
in_keyword = keyword
|
|
else:
|
|
process_macro(keyword, line.strip())
|
|
break
|
|
|
|
if in_keyword:
|
|
process_macro(in_keyword, partial)
|
|
|
|
if extravalues:
|
|
for k,v in list(extravalues.items()):
|
|
if v:
|
|
if v.startswith('$') or v.startswith('@') or v.startswith('%'):
|
|
del extravalues[k]
|
|
else:
|
|
extravalues[k] = v.strip('"\'').rstrip('()')
|
|
|
|
if unmapped:
|
|
outlines.append('# NOTE: the following prog dependencies are unknown, ignoring: %s' % ' '.join(list(set(unmapped))))
|
|
|
|
RecipeHandler.handle_depends(libdeps, pcdeps, deps, outlines, values, tinfoil.config_data)
|
|
|
|
for handler in handlers:
|
|
handler.post_process(srctree, libdeps, pcdeps, deps, outlines, inherits, values)
|
|
|
|
if inherits:
|
|
values['inherit'] = ' '.join(list(set(inherits)))
|
|
|
|
return values
|
|
|
|
|
|
class AutotoolsExtensionHandler(object):
|
|
'''Base class for Autotools extension handlers'''
|
|
def process_macro(self, srctree, keyword, value, process_value, libdeps, pcdeps, deps, outlines, inherits, values):
|
|
'''
|
|
Handle a macro parsed out of an autotools file. Note that if you want this to be called
|
|
for any macro other than the ones AutotoolsRecipeHandler already looks for, you'll need
|
|
to add it to the keywords list in extend_keywords().
|
|
Return True if you've completely handled the passed in macro, otherwise return False.
|
|
'''
|
|
return False
|
|
|
|
def extend_keywords(self, keywords):
|
|
'''Adds keywords to be recognised by the parser (so that you get a call to process_macro)'''
|
|
return
|
|
|
|
def process_prog(self, srctree, keyword, value, prog, deps, outlines, inherits, values):
|
|
'''
|
|
Handle an AC_PATH_PROG, AC_CHECK_PROG etc. line
|
|
Return True if you've completely handled the passed in macro, otherwise return False.
|
|
'''
|
|
return False
|
|
|
|
def post_process(self, srctree, fn, pkg, deps, outlines, inherits, values):
|
|
'''
|
|
Apply any desired post-processing on the output
|
|
'''
|
|
return
|
|
|
|
|
|
class MakefileRecipeHandler(RecipeHandler):
|
|
def process(self, srctree, classes, lines_before, lines_after, handled, extravalues):
|
|
if 'buildsystem' in handled:
|
|
return False
|
|
|
|
makefile = RecipeHandler.checkfiles(srctree, ['Makefile', 'makefile', 'GNUmakefile'])
|
|
if makefile:
|
|
lines_after.append('# NOTE: this is a Makefile-only piece of software, so we cannot generate much of the')
|
|
lines_after.append('# recipe automatically - you will need to examine the Makefile yourself and ensure')
|
|
lines_after.append('# that the appropriate arguments are passed in.')
|
|
lines_after.append('')
|
|
|
|
scanfile = os.path.join(srctree, 'configure.scan')
|
|
skipscan = False
|
|
try:
|
|
stdout, stderr = bb.process.run('autoscan', cwd=srctree, shell=True)
|
|
except bb.process.ExecutionError as e:
|
|
skipscan = True
|
|
if scanfile and os.path.exists(scanfile):
|
|
values = AutotoolsRecipeHandler.extract_autotools_deps(lines_before, srctree, acfile=scanfile)
|
|
classes.extend(values.pop('inherit', '').split())
|
|
for var, value in values.items():
|
|
if var == 'DEPENDS':
|
|
lines_before.append('# NOTE: some of these dependencies may be optional, check the Makefile and/or upstream documentation')
|
|
lines_before.append('%s = "%s"' % (var, value))
|
|
lines_before.append('')
|
|
for f in ['configure.scan', 'autoscan.log']:
|
|
fp = os.path.join(srctree, f)
|
|
if os.path.exists(fp):
|
|
os.remove(fp)
|
|
|
|
self.genfunction(lines_after, 'do_configure', ['# Specify any needed configure commands here'])
|
|
|
|
func = []
|
|
func.append('# You will almost certainly need to add additional arguments here')
|
|
func.append('oe_runmake')
|
|
self.genfunction(lines_after, 'do_compile', func)
|
|
|
|
installtarget = True
|
|
try:
|
|
stdout, stderr = bb.process.run('make -n install', cwd=srctree, shell=True)
|
|
except bb.process.ExecutionError as e:
|
|
if e.exitcode != 1:
|
|
installtarget = False
|
|
func = []
|
|
if installtarget:
|
|
func.append('# This is a guess; additional arguments may be required')
|
|
makeargs = ''
|
|
with open(makefile[0], 'r', errors='surrogateescape') as f:
|
|
for i in range(1, 100):
|
|
if 'DESTDIR' in f.readline():
|
|
makeargs += " 'DESTDIR=${D}'"
|
|
break
|
|
func.append('oe_runmake install%s' % makeargs)
|
|
else:
|
|
func.append('# NOTE: unable to determine what to put here - there is a Makefile but no')
|
|
func.append('# target named "install", so you will need to define this yourself')
|
|
self.genfunction(lines_after, 'do_install', func)
|
|
|
|
handled.append('buildsystem')
|
|
else:
|
|
lines_after.append('# NOTE: no Makefile found, unable to determine what needs to be done')
|
|
lines_after.append('')
|
|
self.genfunction(lines_after, 'do_configure', ['# Specify any needed configure commands here'])
|
|
self.genfunction(lines_after, 'do_compile', ['# Specify compilation commands here'])
|
|
self.genfunction(lines_after, 'do_install', ['# Specify install commands here'])
|
|
|
|
|
|
class VersionFileRecipeHandler(RecipeHandler):
|
|
def process(self, srctree, classes, lines_before, lines_after, handled, extravalues):
|
|
if 'PV' not in extravalues:
|
|
# Look for a VERSION or version file containing a single line consisting
|
|
# only of a version number
|
|
filelist = RecipeHandler.checkfiles(srctree, ['VERSION', 'version'])
|
|
version = None
|
|
for fileitem in filelist:
|
|
linecount = 0
|
|
with open(fileitem, 'r', errors='surrogateescape') as f:
|
|
for line in f:
|
|
line = line.rstrip().strip('"\'')
|
|
linecount += 1
|
|
if line:
|
|
if linecount > 1:
|
|
version = None
|
|
break
|
|
else:
|
|
if validate_pv(line):
|
|
version = line
|
|
if version:
|
|
extravalues['PV'] = version
|
|
break
|
|
|
|
|
|
class SpecFileRecipeHandler(RecipeHandler):
|
|
def process(self, srctree, classes, lines_before, lines_after, handled, extravalues):
|
|
if 'PV' in extravalues and 'PN' in extravalues:
|
|
return
|
|
filelist = RecipeHandler.checkfiles(srctree, ['*.spec'], recursive=True)
|
|
valuemap = {'Name': 'PN',
|
|
'Version': 'PV',
|
|
'Summary': 'SUMMARY',
|
|
'Url': 'HOMEPAGE',
|
|
'License': 'LICENSE'}
|
|
foundvalues = {}
|
|
for fileitem in filelist:
|
|
linecount = 0
|
|
with open(fileitem, 'r', errors='surrogateescape') as f:
|
|
for line in f:
|
|
for value, varname in valuemap.items():
|
|
if line.startswith(value + ':') and not varname in foundvalues:
|
|
foundvalues[varname] = line.split(':', 1)[1].strip()
|
|
break
|
|
if len(foundvalues) == len(valuemap):
|
|
break
|
|
# Drop values containing unexpanded RPM macros
|
|
for k in list(foundvalues.keys()):
|
|
if '%' in foundvalues[k]:
|
|
del foundvalues[k]
|
|
if 'PV' in foundvalues:
|
|
if not validate_pv(foundvalues['PV']):
|
|
del foundvalues['PV']
|
|
license = foundvalues.pop('LICENSE', None)
|
|
if license:
|
|
liccomment = '# NOTE: spec file indicates the license may be "%s"' % license
|
|
for i, line in enumerate(lines_before):
|
|
if line.startswith('LICENSE ='):
|
|
lines_before.insert(i, liccomment)
|
|
break
|
|
else:
|
|
lines_before.append(liccomment)
|
|
extravalues.update(foundvalues)
|
|
|
|
def register_recipe_handlers(handlers):
|
|
# Set priorities with some gaps so that other plugins can insert
|
|
# their own handlers (so avoid changing these numbers)
|
|
handlers.append((CmakeRecipeHandler(), 50))
|
|
handlers.append((AutotoolsRecipeHandler(), 40))
|
|
handlers.append((SconsRecipeHandler(), 30))
|
|
handlers.append((QmakeRecipeHandler(), 20))
|
|
handlers.append((MakefileRecipeHandler(), 10))
|
|
handlers.append((VersionFileRecipeHandler(), -1))
|
|
handlers.append((SpecFileRecipeHandler(), -1))
|