blob: 7c780a52e04f027bc415eab0aa95f219e1fc9347 [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
Oliver Smithadfa0c02021-10-04 11:34:36 +0200113parser.add_argument('-a', '--auto-distclean', action='store_true',
114 help='''run "make distclean" automatically if source directory already configured''')
115
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200116args = parser.parse_args()
117
Neels Hofmeyr450dac72017-08-22 19:27:08 +0200118class listdict(dict):
119 'a dict of lists { "a": [1, 2, 3], "b": [1, 2] }'
120
121 def add(self, name, item):
122 l = self.get(name)
123 if not l:
124 l = []
125 self[name] = l
126 l.append(item)
127
128 def extend(self, name, l):
129 for v in l:
130 self.add(name, v)
131
132 def add_dict(self, d):
133 for k,v in d.items():
134 self.add(k, v)
135
136 def extend_dict(self, d):
137 for k,v in d.items():
138 l = self.extend(k, v)
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200139
140def read_projects_deps(path):
141 'Read deps config and return tuples of (project_name, which-other-to-build-first).'
142 l = []
143 for line in open(path):
144 line = line.strip()
145 if not line or line.startswith('#'):
146 continue
147 tokens = line.split()
148 l.append((tokens[0], tokens[1:]))
149 return l
150
151def read_configure_opts(path):
152 'Read config opts file and return tuples of (project_name, config-opts).'
153 if not path:
154 return {}
155 return dict(read_projects_deps(path))
156
Neels Hofmeyrf6078c42018-09-04 14:34:33 +0200157def 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 +0200158 src_proj = os.path.join(src_dir, proj)
Neels Hofmeyr29fde6f2018-04-01 15:57:02 +0200159 if proj == 'openbsc':
160 src_proj = os.path.join(src_proj, 'openbsc')
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200161 build_proj = os.path.join(build_dir, proj)
162
163 make_to_src = os.path.relpath(src_dir, make_dir)
164 make_to_src_proj = os.path.relpath(src_proj, make_dir)
165 make_to_build_proj = os.path.relpath(build_proj, make_dir)
166 build_to_src = os.path.relpath(src_proj, build_proj)
167
168 if configure_opts:
169 configure_opts_str = ' '.join(configure_opts)
170 else:
171 configure_opts_str = ''
172
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200173 return r'''
174### {proj} ###
175
Oliver Smith505f60d2021-10-04 12:55:44 +0200176{proj}_configure_files := $(shell find {src_proj} \
177 -name "Makefile.am" \
178 -or -name "*.in" \
179 -and -not -name "Makefile.in" \
180 -and -not -name "config.h.in" )
181{proj}_files := $(shell find {src_proj} \
Oliver Smithf7f8c962021-10-04 12:57:26 +0200182 \( \
183 -name "*.[hc]" \
184 -or -name "*.py" \
185 -or -name "*.cpp" \
186 -or -name "*.tpl" \
187 -or -name "*.map" \
188 \) \
189 -and -not -name "config.h")
Neels Hofmeyr1c1e4d22017-09-11 01:32:50 +0200190
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200191.make.{proj}.clone:
Vadim Yanitskiyd9cdec62020-06-01 00:52:12 +0700192 @echo -e "\n\n\n===== $@\n"
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200193 test -d {src} || mkdir -p {src}
Neels Hofmeyr28d4be52018-03-16 03:44:07 +0100194 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 +0200195 sync
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200196 touch $@
197
198.make.{proj}.autoconf: .make.{proj}.clone {src_proj}/configure.ac
Oliver Smithadfa0c02021-10-04 11:34:36 +0200199 if {distclean_cond}; then $(MAKE) {proj}-distclean; fi
Vadim Yanitskiyd9cdec62020-06-01 00:52:12 +0700200 @echo -e "\n\n\n===== $@\n"
Neels Hofmeyr1fa9b292018-04-23 17:04:25 +0200201 -rm -f {src_proj}/.version
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200202 cd {src_proj}; autoreconf -fi
Neels Hofmeyr7bd5c312018-07-26 16:43:01 +0200203 sync
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200204 touch $@
205
Neels Hofmeyr367cb972017-12-15 04:01:13 +0100206.make.{proj}.configure: .make.{proj}.autoconf {deps_installed} $({proj}_configure_files)
Oliver Smithadfa0c02021-10-04 11:34:36 +0200207 if {distclean_cond}; then $(MAKE) {proj}-distclean .make.{proj}.autoconf; fi
Vadim Yanitskiyd9cdec62020-06-01 00:52:12 +0700208 @echo -e "\n\n\n===== $@\n"
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200209 -chmod -R ug+w {build_proj}
210 -rm -rf {build_proj}
211 mkdir -p {build_proj}
Neels Hofmeyr2325e2e2021-10-23 20:42:44 +0200212 cd {build_proj}; {cflags}{docker_cmd}{build_to_src}/configure {configure_opts}
Neels Hofmeyr7bd5c312018-07-26 16:43:01 +0200213 sync
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200214 touch $@
215
Neels Hofmeyr367cb972017-12-15 04:01:13 +0100216.make.{proj}.build: .make.{proj}.configure $({proj}_files)
Oliver Smithadfa0c02021-10-04 11:34:36 +0200217 if {distclean_cond}; then $(MAKE) {proj}-distclean .make.{proj}.configure; fi
Vadim Yanitskiyd9cdec62020-06-01 00:52:12 +0700218 @echo -e "\n\n\n===== $@\n"
Oliver Smithc47eafb2021-08-12 12:45:36 +0200219 {docker_cmd}$(MAKE) -C {build_proj} -j {jobs} {check}
Neels Hofmeyr7bd5c312018-07-26 16:43:01 +0200220 sync
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200221 touch $@
222
223.make.{proj}.install: .make.{proj}.build
Vadim Yanitskiyd9cdec62020-06-01 00:52:12 +0700224 @echo -e "\n\n\n===== $@\n"
Oliver Smithc47eafb2021-08-12 12:45:36 +0200225 {docker_cmd}{sudo_make_install}$(MAKE) -C {build_proj} install
Neels Hofmeyr461c3bd2017-12-06 00:32:15 +0100226 {no_ldconfig}{sudo_ldconfig}ldconfig
Neels Hofmeyr7bd5c312018-07-26 16:43:01 +0200227 sync
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200228 touch $@
Neels Hofmeyr277f4792017-08-29 12:30:32 +0200229
Neels Hofmeyr6d16ae92018-09-04 14:36:15 +0200230.PHONY: {proj}
Neels Hofmeyr3f934122017-08-29 12:31:59 +0200231{proj}: .make.{proj}.install
232
Neels Hofmeyr4d38c1e2018-04-23 17:04:15 +0200233.PHONY: {proj}-reinstall
234{proj}-reinstall: {deps_reinstall}
235 {sudo_make_install}$(MAKE) -C {build_proj} install
236
Neels Hofmeyr277f4792017-08-29 12:30:32 +0200237.PHONY: {proj}-clean
238{proj}-clean:
Vadim Yanitskiyd9cdec62020-06-01 00:52:12 +0700239 @echo -e "\n\n\n===== $@\n"
Neels Hofmeyr277f4792017-08-29 12:30:32 +0200240 -chmod -R ug+w {build_proj}
241 -rm -rf {build_proj}
242 -rm -rf .make.{proj}.*
Oliver Smithadfa0c02021-10-04 11:34:36 +0200243
244.PHONY: {proj}-distclean
245{proj}-distclean: {proj}-clean
246 @echo -e "\n\n\n===== $@\n"
247 $(MAKE) -C {src_proj} distclean
248
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200249'''.format(
250 url=url,
Neels Hofmeyr28d4be52018-03-16 03:44:07 +0100251 push_url=push_url or url,
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200252 proj=proj,
253 jobs=jobs,
254 src=make_to_src,
255 src_proj=make_to_src_proj,
256 build_proj=make_to_build_proj,
257 build_to_src=build_to_src,
258 deps_installed=' '.join(['.make.%s.install' % d for d in deps]),
Neels Hofmeyr4d38c1e2018-04-23 17:04:15 +0200259 deps_reinstall=' '.join(['%s-reinstall' %d for d in deps]),
Neels Hofmeyr461c3bd2017-12-06 00:32:15 +0100260 configure_opts=configure_opts_str,
261 sudo_make_install='sudo ' if sudo_make_install else '',
262 no_ldconfig='#' if no_ldconfig else '',
Neels Hofmeyrf6078c42018-09-04 14:34:33 +0200263 sudo_ldconfig='' if ldconfig_without_sudo else 'sudo ',
264 check='check' if make_check else '',
Oliver Smithc47eafb2021-08-12 12:45:36 +0200265 docker_cmd=f'{args.docker_cmd} ' if args.docker_cmd else '',
Neels Hofmeyr2325e2e2021-10-23 20:42:44 +0200266 cflags='CFLAGS=-g ' if args.build_debug else '',
Oliver Smithadfa0c02021-10-04 11:34:36 +0200267 distclean_cond=f'[ -e {make_to_src_proj}/config.status ]' if args.auto_distclean else 'false'
Neels Hofmeyr461c3bd2017-12-06 00:32:15 +0100268 )
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200269
270
Oliver Smith0a4d8ea2021-07-29 14:32:29 +0200271projects_deps = read_projects_deps(all_deps_file)
Neels Hofmeyr450dac72017-08-22 19:27:08 +0200272configure_opts = listdict()
273configure_opts_files = sorted(args.configure_opts_files or [])
274for configure_opts_file in configure_opts_files:
Oliver Smith0a4d8ea2021-07-29 14:32:29 +0200275 if configure_opts_file.endswith(".deps"):
276 print(f"WARNING: using {all_deps_file} instead of {configure_opts_file}")
277 continue
Neels Hofmeyr450dac72017-08-22 19:27:08 +0200278 r = read_configure_opts(configure_opts_file)
279 configure_opts.extend_dict(read_configure_opts(configure_opts_file))
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200280
281make_dir = args.make_dir
282if not make_dir:
Neels Hofmeyr450dac72017-08-22 19:27:08 +0200283 opts_names = '+'.join([f.replace('.opts', '') for f in configure_opts_files])
Oliver Smith0a4d8ea2021-07-29 14:32:29 +0200284 make_dir = 'make-%s' % opts_names
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200285
286if not os.path.isdir(make_dir):
287 os.makedirs(make_dir)
288
289build_dir = args.build_dir
290if not build_dir:
291 build_dir = make_dir
292
293output = os.path.join(make_dir, args.output)
294print('Writing to %r' % output)
295
296with open(output, 'w') as out:
297 out.write('# This Makefile was generated by %s\n' % os.path.basename(sys.argv[0]))
298
Oliver Smith0a4d8ea2021-07-29 14:32:29 +0200299 configure_opts_args = ""
300 for f in configure_opts_files:
301 if not f.endswith(".deps"):
302 configure_opts_args += f' \\\n\t\t{os.path.relpath(f, make_dir)}'
303
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200304 # convenience: add a regen target that updates the generated makefile itself
305 out.write(r'''
Oliver Smith0a4d8ea2021-07-29 14:32:29 +0200306default: usrp
307
308cn: \
309 osmo-ggsn \
310 osmo-hlr \
311 osmo-iuh \
312 osmo-mgw \
313 osmo-msc \
314 osmo-sgsn \
315 osmo-sip-connector \
316 osmo-smlc \
317 $(NULL)
318
319cn-bsc: \
320 cn \
321 osmo-bsc \
322 $(NULL)
323
324usrp: \
325 cn-bsc \
326 osmo-bts \
327 osmo-trx \
328 $(NULL)
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200329
Neels Hofmeyr1b0d34f2018-03-16 03:46:08 +0100330.PHONY: all_debug
331all_debug:
332 $(MAKE) --dry-run -d all | grep "is newer than target"
333 $(MAKE) all
334
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200335# regenerate this Makefile, in case the deps or opts changed
336.PHONY: regen
337regen:
Oliver Smithb3ae4b62021-01-28 11:20:04 +0100338 {script} \
Oliver Smithb3ae4b62021-01-28 11:20:04 +0100339 {configure_opts} \
340 -m {make_dir} \
341 -o {makefile} \
342 -s {src_dir} \
343 -b {build_dir} \
Oliver Smithadfa0c02021-10-04 11:34:36 +0200344 -u "{url}"{push_url}{sudo_make_install}{no_ldconfig}{ldconfig_without_sudo}{make_check}{docker_cmd}{build_debug}{auto_distclean}
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200345
346'''.format(
347 script=os.path.relpath(sys.argv[0], make_dir),
Oliver Smith0a4d8ea2021-07-29 14:32:29 +0200348 configure_opts=configure_opts_args,
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200349 make_dir='.',
350 makefile=args.output,
351 src_dir=os.path.relpath(args.src_dir, make_dir),
352 build_dir=os.path.relpath(build_dir, make_dir),
Neels Hofmeyr461c3bd2017-12-06 00:32:15 +0100353 url=args.url,
Oliver Smithb3ae4b62021-01-28 11:20:04 +0100354 push_url=(" \\\n\t\t-p '%s'"%args.push_url) if args.push_url else '',
355 sudo_make_install=' \\\n\t\t-I' if args.sudo_make_install else '',
356 no_ldconfig=' \\\n\t\t-L' if args.no_ldconfig else '',
357 ldconfig_without_sudo=' \\\n\t\t--ldconfig-without-sudo' if args.ldconfig_without_sudo else '',
358 make_check='' if args.make_check else " \\\n\t\t--no-make-check",
Neels Hofmeyr2325e2e2021-10-23 20:42:44 +0200359 docker_cmd=f' \\\n\t\t--docker-cmd "{args.docker_cmd}"' if args.docker_cmd else '',
360 build_debug=f' \\\n\t\t--build-debug' if args.build_debug else '',
Oliver Smithadfa0c02021-10-04 11:34:36 +0200361 auto_distclean=' \\\n\t\t--auto-distclean' if args.auto_distclean else '',
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200362 ))
363
Neels Hofmeyre274d352017-08-22 17:31:03 +0200364 # convenience target: clone all repositories first
365 out.write('clone: \\\n\t' + ' \\\n\t'.join([ '.make.%s.clone' % p for p, d in projects_deps ]) + '\n\n')
366
Neels Hofmeyr277f4792017-08-29 12:30:32 +0200367 # convenience target: clean all
368 out.write('clean: \\\n\t' + ' \\\n\t'.join([ '%s-clean' % p for p, d in projects_deps ]) + '\n\n')
369
Neels Hofmeyr0a1bdff2017-08-13 03:22:42 +0200370 # now the actual useful build rules
Neels Hofmeyr1c1e4d22017-09-11 01:32:50 +0200371 out.write('all: clone all-install\n\n')
372
373 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 +0200374
375 for proj, deps in projects_deps:
Neels Hofmeyra007eaa2018-09-04 14:37:13 +0200376 all_config_opts = []
377 all_config_opts.extend(configure_opts.get('ALL') or [])
378 all_config_opts.extend(configure_opts.get(proj) or [])
379 out.write(gen_make(proj, deps, all_config_opts, args.jobs,
Neels Hofmeyr28d4be52018-03-16 03:44:07 +0100380 make_dir, args.src_dir, build_dir, args.url, args.push_url,
Neels Hofmeyr461c3bd2017-12-06 00:32:15 +0100381 args.sudo_make_install, args.no_ldconfig,
Neels Hofmeyrf6078c42018-09-04 14:34:33 +0200382 args.ldconfig_without_sudo, args.make_check))
Neels Hofmeyrd4d88482017-08-22 19:27:28 +0200383
384# vim: expandtab tabstop=2 shiftwidth=2