blob: 90a7476e11d919c1348086bf9da2c860f38a163e [file] [log] [blame]
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +02001#!/usr/bin/env python3
2'''
3Generate a top-level makefile that builds the Osmocom 2G + 3G network components.
4
Oliver Smith0a4d8ea2021-07-29 14:32:29 +02005 ./gen_makefile.py [configure.opts [more.opts]] [-o Makefile.output]
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +02006
7Configured by text files:
8
Oliver Smith0a4d8ea2021-07-29 14:32:29 +02009 all.deps: whitespace-separated listing of
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +020010 project_name depends_on_project_1 depends_on_project_2 ...
11
12 *.opts: whitespace-separated listing of
13 project_name --config-opt-1 --config-opt-2 ...
14
15Thus it is possible to choose between e.g.
Oliver Smith0a4d8ea2021-07-29 14:32:29 +020016- building each of those with or without mgcp transcoding support by adding or
17 removing "transcoding.opts" from the command line
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +020018
19From the Makefile nature, the dependencies extend, no need to repeat common deps.
20
21When this script is done, a Makefile has been generated that allows you to
22build all projects at once by issuing 'make', but also to refresh only parts of
23it when some bits in the middle have changed. The makefile keeps local progress
24marker files like .make.libosmocore.configure; if such progress marker is
25removed or becomes outdated, that step and all dependent ones are re-run.
26This is helpful in daily hacking across several repositories.
Neels Hofmeyr461c3bd2017-12-06 00:32:15 +010027
28Note that by default, this includes 'sudo ldconfig' calls following each
29installation. You may want to permit your user to run 'sudo ldconfig' without
30needing a password, e.g. by
31
32 sudo sh -c "echo '$USER ALL= NOPASSWD: /sbin/ldconfig' > /etc/sudoers.d/${USER}_ldconfig"
33
34You can skip the 'sudo ldconfig' by issuing the --no-ldconfig option.
35
36You can run 'ldconfig' without sudo by issuing the --ldconfig-without-sudo option.
37
38By default, it is assumed that your user has write permission to /usr/local. If you
39need sudo to install there, you may issue the --sudo-make-install option.
Neels Hofmeyr2535a262018-01-16 16:34:32 +010040
41EXAMPLE:
42
Oliver Smith0a4d8ea2021-07-29 14:32:29 +020043 ./gen_makefile.py default.opts iu.opts -I -m build
Neels Hofmeyr2535a262018-01-16 16:34:32 +010044 cd build
45 make
46
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +020047'''
48
49import sys
50import os
51import argparse
52
Oliver Smith0a4d8ea2021-07-29 14:32:29 +020053topdir = os.path.dirname(os.path.realpath(__file__))
54all_deps_file = os.path.join(topdir, "all.deps")
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +020055parser = argparse.ArgumentParser(epilog=__doc__, formatter_class=argparse.RawTextHelpFormatter)
56
Neels Hofmeyr450dac72017-08-22 19:27:08 +020057parser.add_argument('configure_opts_files',
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +020058 help='''Config file containing project name and
59./configure options''',
Neels Hofmeyr450dac72017-08-22 19:27:08 +020060 nargs='*')
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +020061
62parser.add_argument('-m', '--make-dir', dest='make_dir',
63 help='''Place Makefile in this dir (default: create
Oliver Smith0a4d8ea2021-07-29 14:32:29 +020064a new dir named after opts files).''')
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +020065
66parser.add_argument('-s', '--src-dir', dest='src_dir', default='./src',
67 help='Parent dir for all git clones.')
68
69parser.add_argument('-b', '--build-dir', dest='build_dir',
70 help='''Parent dir for all build trees (default:
71directly in the make-dir).''')
72
Neels Hofmeyrbffdc302017-12-06 00:31:49 +010073parser.add_argument('-u', '--url', dest='url', default='git://git.osmocom.org',
74 help='''git clone base URL. Default is 'git://git.osmocom.org'.
75e.g. with a config like this in your ~/.ssh/config:
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +020076 host go
77 hostname gerrit.osmocom.org
78 port 29418
Neels Hofmeyrbffdc302017-12-06 00:31:49 +010079you may pass '-u ssh://go' to be able to submit to gerrit.''')
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +020080
Neels Hofmeyr972c2942018-03-16 03:49:58 +010081parser.add_argument('-p', '--push-url', dest='push_url', default='',
Neels Hofmeyr28d4be52018-03-16 03:44:07 +010082 help='''git push-URL. Default is to not configure a separate push-URL.''')
83
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +020084parser.add_argument('-o', '--output', dest='output', default='Makefile',
85 help='''Makefile filename (default: 'Makefile').''')
86
87parser.add_argument('-j', '--jobs', dest='jobs', default='9',
88 help='''-j option to pass to 'make'.''')
89
Neels Hofmeyr461c3bd2017-12-06 00:32:15 +010090parser.add_argument('-I', '--sudo-make-install', dest='sudo_make_install',
91 action='store_true',
92 help='''run 'make install' step with 'sudo'.''')
93
94parser.add_argument('-L', '--no-ldconfig', dest='no_ldconfig',
95 action='store_true',
96 help='''omit the 'sudo ldconfig' step.''')
97
98parser.add_argument('--ldconfig-without-sudo', dest='ldconfig_without_sudo',
99 action='store_true',
100 help='''call just 'ldconfig', without sudo, which implies
101root privileges (not recommended)''')
102
Neels Hofmeyrf6078c42018-09-04 14:34:33 +0200103parser.add_argument('-c', '--no-make-check', dest='make_check',
104 default=True, action='store_false',
105 help='''do not 'make check', just 'make' to build.''')
106
Oliver Smithc47eafb2021-08-12 12:45:36 +0200107parser.add_argument('--docker-cmd',
108 help='''prefix configure/make/make install calls with this command (used by ttcn3.sh)''')
109
Neels Hofmeyr2325e2e2021-10-23 20:42:44 +0200110parser.add_argument('-g', '--build-debug', dest='build_debug', default=False, action='store_true',
111 help='''set 'CFLAGS=-g' when calling src/configure''')
112
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200113args = parser.parse_args()
114
Neels Hofmeyr450dac72017-08-22 19:27:08 +0200115class listdict(dict):
116 'a dict of lists { "a": [1, 2, 3], "b": [1, 2] }'
117
118 def add(self, name, item):
119 l = self.get(name)
120 if not l:
121 l = []
122 self[name] = l
123 l.append(item)
124
125 def extend(self, name, l):
126 for v in l:
127 self.add(name, v)
128
129 def add_dict(self, d):
130 for k,v in d.items():
131 self.add(k, v)
132
133 def extend_dict(self, d):
134 for k,v in d.items():
135 l = self.extend(k, v)
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200136
137def read_projects_deps(path):
138 'Read deps config and return tuples of (project_name, which-other-to-build-first).'
139 l = []
140 for line in open(path):
141 line = line.strip()
142 if not line or line.startswith('#'):
143 continue
144 tokens = line.split()
145 l.append((tokens[0], tokens[1:]))
146 return l
147
148def read_configure_opts(path):
149 'Read config opts file and return tuples of (project_name, config-opts).'
150 if not path:
151 return {}
152 return dict(read_projects_deps(path))
153
Neels Hofmeyrf6078c42018-09-04 14:34:33 +0200154def gen_make(proj, deps, configure_opts, jobs, make_dir, src_dir, build_dir, url, push_url, sudo_make_install, no_ldconfig, ldconfig_without_sudo, make_check):
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200155 src_proj = os.path.join(src_dir, proj)
Neels Hofmeyr29fde6f2018-04-01 15:57:02 +0200156 if proj == 'openbsc':
157 src_proj = os.path.join(src_proj, 'openbsc')
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200158 build_proj = os.path.join(build_dir, proj)
159
160 make_to_src = os.path.relpath(src_dir, make_dir)
161 make_to_src_proj = os.path.relpath(src_proj, make_dir)
162 make_to_build_proj = os.path.relpath(build_proj, make_dir)
163 build_to_src = os.path.relpath(src_proj, build_proj)
164
165 if configure_opts:
166 configure_opts_str = ' '.join(configure_opts)
167 else:
168 configure_opts_str = ''
169
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200170 return r'''
171### {proj} ###
172
Oliver Smith505f60d2021-10-04 12:55:44 +0200173{proj}_configure_files := $(shell find {src_proj} \
174 -name "Makefile.am" \
175 -or -name "*.in" \
176 -and -not -name "Makefile.in" \
177 -and -not -name "config.h.in" )
178{proj}_files := $(shell find {src_proj} \
179 -name "*.[hc]" \
180 -or -name "*.py" \
181 -or -name "*.cpp" \
182 -or -name "*.tpl" \
183 -or -name "*.map")
Neels Hofmeyr1c1e4d22017-09-11 01:32:50 +0200184
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200185.make.{proj}.clone:
Vadim Yanitskiyd9cdec62020-06-01 00:52:12 +0700186 @echo -e "\n\n\n===== $@\n"
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200187 test -d {src} || mkdir -p {src}
Neels Hofmeyr28d4be52018-03-16 03:44:07 +0100188 test -d {src_proj} || ( git -C {src} clone "{url}/{proj}" "{proj}" && git -C "{src}/{proj}" remote set-url --push origin "{push_url}/{proj}" )
Neels Hofmeyr7bd5c312018-07-26 16:43:01 +0200189 sync
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200190 touch $@
191
192.make.{proj}.autoconf: .make.{proj}.clone {src_proj}/configure.ac
Vadim Yanitskiyd9cdec62020-06-01 00:52:12 +0700193 @echo -e "\n\n\n===== $@\n"
Neels Hofmeyr1fa9b292018-04-23 17:04:25 +0200194 -rm -f {src_proj}/.version
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200195 cd {src_proj}; autoreconf -fi
Neels Hofmeyr7bd5c312018-07-26 16:43:01 +0200196 sync
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200197 touch $@
198
Neels Hofmeyr367cb972017-12-15 04:01:13 +0100199.make.{proj}.configure: .make.{proj}.autoconf {deps_installed} $({proj}_configure_files)
Vadim Yanitskiyd9cdec62020-06-01 00:52:12 +0700200 @echo -e "\n\n\n===== $@\n"
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200201 -chmod -R ug+w {build_proj}
202 -rm -rf {build_proj}
203 mkdir -p {build_proj}
Neels Hofmeyr2325e2e2021-10-23 20:42:44 +0200204 cd {build_proj}; {cflags}{docker_cmd}{build_to_src}/configure {configure_opts}
Neels Hofmeyr7bd5c312018-07-26 16:43:01 +0200205 sync
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200206 touch $@
207
Neels Hofmeyr367cb972017-12-15 04:01:13 +0100208.make.{proj}.build: .make.{proj}.configure $({proj}_files)
Vadim Yanitskiyd9cdec62020-06-01 00:52:12 +0700209 @echo -e "\n\n\n===== $@\n"
Oliver Smithc47eafb2021-08-12 12:45:36 +0200210 {docker_cmd}$(MAKE) -C {build_proj} -j {jobs} {check}
Neels Hofmeyr7bd5c312018-07-26 16:43:01 +0200211 sync
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200212 touch $@
213
214.make.{proj}.install: .make.{proj}.build
Vadim Yanitskiyd9cdec62020-06-01 00:52:12 +0700215 @echo -e "\n\n\n===== $@\n"
Oliver Smithc47eafb2021-08-12 12:45:36 +0200216 {docker_cmd}{sudo_make_install}$(MAKE) -C {build_proj} install
Neels Hofmeyr461c3bd2017-12-06 00:32:15 +0100217 {no_ldconfig}{sudo_ldconfig}ldconfig
Neels Hofmeyr7bd5c312018-07-26 16:43:01 +0200218 sync
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200219 touch $@
Neels Hofmeyr277f4792017-08-29 12:30:32 +0200220
Neels Hofmeyr6d16ae92018-09-04 14:36:15 +0200221.PHONY: {proj}
Neels Hofmeyr3f934122017-08-29 12:31:59 +0200222{proj}: .make.{proj}.install
223
Neels Hofmeyr4d38c1e2018-04-23 17:04:15 +0200224.PHONY: {proj}-reinstall
225{proj}-reinstall: {deps_reinstall}
226 {sudo_make_install}$(MAKE) -C {build_proj} install
227
Neels Hofmeyr277f4792017-08-29 12:30:32 +0200228.PHONY: {proj}-clean
229{proj}-clean:
Vadim Yanitskiyd9cdec62020-06-01 00:52:12 +0700230 @echo -e "\n\n\n===== $@\n"
Neels Hofmeyr277f4792017-08-29 12:30:32 +0200231 -chmod -R ug+w {build_proj}
232 -rm -rf {build_proj}
233 -rm -rf .make.{proj}.*
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200234'''.format(
235 url=url,
Neels Hofmeyr28d4be52018-03-16 03:44:07 +0100236 push_url=push_url or url,
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200237 proj=proj,
238 jobs=jobs,
239 src=make_to_src,
240 src_proj=make_to_src_proj,
241 build_proj=make_to_build_proj,
242 build_to_src=build_to_src,
243 deps_installed=' '.join(['.make.%s.install' % d for d in deps]),
Neels Hofmeyr4d38c1e2018-04-23 17:04:15 +0200244 deps_reinstall=' '.join(['%s-reinstall' %d for d in deps]),
Neels Hofmeyr461c3bd2017-12-06 00:32:15 +0100245 configure_opts=configure_opts_str,
246 sudo_make_install='sudo ' if sudo_make_install else '',
247 no_ldconfig='#' if no_ldconfig else '',
Neels Hofmeyrf6078c42018-09-04 14:34:33 +0200248 sudo_ldconfig='' if ldconfig_without_sudo else 'sudo ',
249 check='check' if make_check else '',
Oliver Smithc47eafb2021-08-12 12:45:36 +0200250 docker_cmd=f'{args.docker_cmd} ' if args.docker_cmd else '',
Neels Hofmeyr2325e2e2021-10-23 20:42:44 +0200251 cflags='CFLAGS=-g ' if args.build_debug else '',
Neels Hofmeyr461c3bd2017-12-06 00:32:15 +0100252 )
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200253
254
Oliver Smith0a4d8ea2021-07-29 14:32:29 +0200255projects_deps = read_projects_deps(all_deps_file)
Neels Hofmeyr450dac72017-08-22 19:27:08 +0200256configure_opts = listdict()
257configure_opts_files = sorted(args.configure_opts_files or [])
258for configure_opts_file in configure_opts_files:
Oliver Smith0a4d8ea2021-07-29 14:32:29 +0200259 if configure_opts_file.endswith(".deps"):
260 print(f"WARNING: using {all_deps_file} instead of {configure_opts_file}")
261 continue
Neels Hofmeyr450dac72017-08-22 19:27:08 +0200262 r = read_configure_opts(configure_opts_file)
263 configure_opts.extend_dict(read_configure_opts(configure_opts_file))
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200264
265make_dir = args.make_dir
266if not make_dir:
Neels Hofmeyr450dac72017-08-22 19:27:08 +0200267 opts_names = '+'.join([f.replace('.opts', '') for f in configure_opts_files])
Oliver Smith0a4d8ea2021-07-29 14:32:29 +0200268 make_dir = 'make-%s' % opts_names
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200269
270if not os.path.isdir(make_dir):
271 os.makedirs(make_dir)
272
273build_dir = args.build_dir
274if not build_dir:
275 build_dir = make_dir
276
277output = os.path.join(make_dir, args.output)
278print('Writing to %r' % output)
279
280with open(output, 'w') as out:
281 out.write('# This Makefile was generated by %s\n' % os.path.basename(sys.argv[0]))
282
Oliver Smith0a4d8ea2021-07-29 14:32:29 +0200283 configure_opts_args = ""
284 for f in configure_opts_files:
285 if not f.endswith(".deps"):
286 configure_opts_args += f' \\\n\t\t{os.path.relpath(f, make_dir)}'
287
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200288 # convenience: add a regen target that updates the generated makefile itself
289 out.write(r'''
Oliver Smith0a4d8ea2021-07-29 14:32:29 +0200290default: usrp
291
292cn: \
293 osmo-ggsn \
294 osmo-hlr \
295 osmo-iuh \
296 osmo-mgw \
297 osmo-msc \
298 osmo-sgsn \
299 osmo-sip-connector \
300 osmo-smlc \
301 $(NULL)
302
303cn-bsc: \
304 cn \
305 osmo-bsc \
306 $(NULL)
307
308usrp: \
309 cn-bsc \
310 osmo-bts \
311 osmo-trx \
312 $(NULL)
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200313
Neels Hofmeyr1b0d34f2018-03-16 03:46:08 +0100314.PHONY: all_debug
315all_debug:
316 $(MAKE) --dry-run -d all | grep "is newer than target"
317 $(MAKE) all
318
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200319# regenerate this Makefile, in case the deps or opts changed
320.PHONY: regen
321regen:
Oliver Smithb3ae4b62021-01-28 11:20:04 +0100322 {script} \
Oliver Smithb3ae4b62021-01-28 11:20:04 +0100323 {configure_opts} \
324 -m {make_dir} \
325 -o {makefile} \
326 -s {src_dir} \
327 -b {build_dir} \
Neels Hofmeyr2325e2e2021-10-23 20:42:44 +0200328 -u "{url}"{push_url}{sudo_make_install}{no_ldconfig}{ldconfig_without_sudo}{make_check}{docker_cmd}{build_debug}
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200329
330'''.format(
331 script=os.path.relpath(sys.argv[0], make_dir),
Oliver Smith0a4d8ea2021-07-29 14:32:29 +0200332 configure_opts=configure_opts_args,
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200333 make_dir='.',
334 makefile=args.output,
335 src_dir=os.path.relpath(args.src_dir, make_dir),
336 build_dir=os.path.relpath(build_dir, make_dir),
Neels Hofmeyr461c3bd2017-12-06 00:32:15 +0100337 url=args.url,
Oliver Smithb3ae4b62021-01-28 11:20:04 +0100338 push_url=(" \\\n\t\t-p '%s'"%args.push_url) if args.push_url else '',
339 sudo_make_install=' \\\n\t\t-I' if args.sudo_make_install else '',
340 no_ldconfig=' \\\n\t\t-L' if args.no_ldconfig else '',
341 ldconfig_without_sudo=' \\\n\t\t--ldconfig-without-sudo' if args.ldconfig_without_sudo else '',
342 make_check='' if args.make_check else " \\\n\t\t--no-make-check",
Neels Hofmeyr2325e2e2021-10-23 20:42:44 +0200343 docker_cmd=f' \\\n\t\t--docker-cmd "{args.docker_cmd}"' if args.docker_cmd else '',
344 build_debug=f' \\\n\t\t--build-debug' if args.build_debug else '',
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200345 ))
346
Neels Hofmeyre274d352017-08-22 17:31:03 +0200347 # convenience target: clone all repositories first
348 out.write('clone: \\\n\t' + ' \\\n\t'.join([ '.make.%s.clone' % p for p, d in projects_deps ]) + '\n\n')
349
Neels Hofmeyr277f4792017-08-29 12:30:32 +0200350 # convenience target: clean all
351 out.write('clean: \\\n\t' + ' \\\n\t'.join([ '%s-clean' % p for p, d in projects_deps ]) + '\n\n')
352
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200353 # now the actual useful build rules
Neels Hofmeyr1c1e4d22017-09-11 01:32:50 +0200354 out.write('all: clone all-install\n\n')
355
356 out.write('all-install: \\\n\t' + ' \\\n\t'.join([ '.make.%s.install' % p for p, d in projects_deps ]) + '\n\n')
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200357
358 for proj, deps in projects_deps:
Neels Hofmeyra007eaa2018-09-04 14:37:13 +0200359 all_config_opts = []
360 all_config_opts.extend(configure_opts.get('ALL') or [])
361 all_config_opts.extend(configure_opts.get(proj) or [])
362 out.write(gen_make(proj, deps, all_config_opts, args.jobs,
Neels Hofmeyr28d4be52018-03-16 03:44:07 +0100363 make_dir, args.src_dir, build_dir, args.url, args.push_url,
Neels Hofmeyr461c3bd2017-12-06 00:32:15 +0100364 args.sudo_make_install, args.no_ldconfig,
Neels Hofmeyrf6078c42018-09-04 14:34:33 +0200365 args.ldconfig_without_sudo, args.make_check))
Neels Hofmeyrd4d88482017-08-22 19:27:28 +0200366
367# vim: expandtab tabstop=2 shiftwidth=2