blob: fa666253616f7e24af6795a238af8dd929b80e13 [file] [log] [blame]
Ryana3627c02009-05-27 14:29:551# /usr/bin/env python
Ryan8ddf9302009-09-02 18:19:522import re
Ryan1a126ed2009-04-04 12:50:153import Options
Ryan41d89f62009-07-28 10:29:184import sys, os, shutil
Ryan Dahld979ac92009-10-09 13:00:125from Utils import cmd_output
Ryan1a126ed2009-04-04 12:50:156from os.path import join, dirname, abspath
Ryana4593e32009-04-23 11:18:387from logging import fatal
8
Ryan Dahld979ac92009-10-09 13:00:129cwd = os.getcwd()
Ryan Dahl027829d2009-11-17 13:30:4010VERSION="0.1.18"
Ryan4d921992009-08-26 23:11:1611APPNAME="node.js"
12
Ryan63a9cd32009-04-15 08:08:2813import js2c
14
Ryan1a126ed2009-04-04 12:50:1515srcdir = '.'
16blddir = 'build'
17
18def set_options(opt):
19 # the gcc module provides a --debug-level option
20 opt.tool_options('compiler_cxx')
21 opt.tool_options('compiler_cc')
Ryan4d921992009-08-26 23:11:1622 opt.tool_options('misc')
Ryan29b528c2009-04-23 15:29:3123 opt.add_option( '--debug'
24 , action='store_true'
25 , default=False
26 , help='Build debug variant [Default: False]'
27 , dest='debug'
28 )
Ryan7bad9de2009-06-16 13:47:5729 opt.add_option( '--efence'
30 , action='store_true'
31 , default=False
32 , help='Build with -lefence for debugging [Default: False]'
33 , dest='efence'
34 )
Ryan1a126ed2009-04-04 12:50:1535
Ryan41d89f62009-07-28 10:29:1836def mkdir_p(dir):
37 if not os.path.exists (dir):
38 os.makedirs (dir)
39
Ryan Dahl18da8ff2009-09-28 20:39:0040# Copied from Python 2.6 because 2.4.4 at least is broken by not using
41# mkdirs
42# http://mail.python.org/pipermail/python-bugs-list/2005-January/027118.html
43def copytree(src, dst, symlinks=False, ignore=None):
44 names = os.listdir(src)
45 if ignore is not None:
46 ignored_names = ignore(src, names)
47 else:
48 ignored_names = set()
49
50 os.makedirs(dst)
51 errors = []
52 for name in names:
53 if name in ignored_names:
54 continue
Ryan Dahl53ebe752009-10-08 21:20:1455 srcname = join(src, name)
56 dstname = join(dst, name)
Ryan Dahl18da8ff2009-09-28 20:39:0057 try:
58 if symlinks and os.path.islink(srcname):
59 linkto = os.readlink(srcname)
60 os.symlink(linkto, dstname)
61 elif os.path.isdir(srcname):
62 copytree(srcname, dstname, symlinks, ignore)
63 else:
64 shutil.copy2(srcname, dstname)
65 # XXX What about devices, sockets etc.?
66 except (IOError, os.error), why:
67 errors.append((srcname, dstname, str(why)))
68 # catch the Error from the recursive copytree so that we can
69 # continue with other files
70 except Error, err:
71 errors.extend(err.args[0])
72 try:
73 shutil.copystat(src, dst)
74 except OSError, why:
75 if WindowsError is not None and isinstance(why, WindowsError):
76 # Copying file access times may fail on Windows
77 pass
78 else:
79 errors.extend((src, dst, str(why)))
80 if errors:
81 raise Error, errors
82
Ryan41d89f62009-07-28 10:29:1883def conf_subproject (conf, subdir, command=None):
84 print("---- %s ----" % subdir)
85 src = join(conf.srcdir, subdir)
86 if not os.path.exists (src): fatal("no such subproject " + subdir)
87
88 default_tgt = join(conf.blddir, "default", subdir)
89
90 if not os.path.exists(default_tgt):
Ryan Dahl18da8ff2009-09-28 20:39:0091 copytree(src, default_tgt, True)
Ryan41d89f62009-07-28 10:29:1892
93 if command:
94 if os.system("cd %s && %s" % (default_tgt, command)) != 0:
95 fatal("Configuring %s failed." % (subdir))
96
97 debug_tgt = join(conf.blddir, "debug", subdir)
98
99 if not os.path.exists(debug_tgt):
Ryan Dahl18da8ff2009-09-28 20:39:00100 copytree(default_tgt, debug_tgt, True)
Ryan41d89f62009-07-28 10:29:18101
Ryan1a126ed2009-04-04 12:50:15102def configure(conf):
103 conf.check_tool('compiler_cxx')
104 conf.check_tool('compiler_cc')
Ryana4593e32009-04-23 11:18:38105
Ryan8e7bbf22009-04-23 17:26:56106 conf.env["USE_DEBUG"] = Options.options.debug
Ryan1a126ed2009-04-04 12:50:15107
Ryan2b6d7242009-06-20 13:07:10108 conf.check(lib='dl', uselib_store='DL')
Ryan8152f9c2009-09-01 12:15:29109 conf.env.append_value("CCFLAGS", "-rdynamic")
Ryana97dce72009-08-31 09:14:34110 conf.env.append_value("LINKFLAGS_DL", "-rdynamic")
111
Ryan8152f9c2009-09-01 12:15:29112 #if Options.options.debug:
113 # conf.check(lib='profiler', uselib_store='PROFILER')
Ryan7bad9de2009-06-16 13:47:57114
Ryan8152f9c2009-09-01 12:15:29115 #if Options.options.efence:
116 # conf.check(lib='efence', libpath=['/usr/lib', '/usr/local/lib'], uselib_store='EFENCE')
Ryana3627c02009-05-27 14:29:55117
Ryan Dahl8b62e862009-10-10 09:58:36118 if not conf.check(lib="execinfo", libpath=['/usr/lib', '/usr/local/lib'], uselib_store="EXECINFO"):
119 if sys.platform.startswith("freebsd"):
Ryan7bad9de2009-06-16 13:47:57120 fatal("Install the libexecinfo port from /usr/ports/devel/libexecinfo.")
Ryana3627c02009-05-27 14:29:55121
Rhys Jonesb6dda612009-11-22 02:58:08122 if conf.check_cfg(package='gnutls',
123 args='--cflags --libs',
124 #libpath=['/usr/lib', '/usr/local/lib'],
125 uselib_store='GNUTLS'):
126 if conf.check(lib='gpg-error',
127 #libpath=['/usr/lib', '/usr/local/lib'],
128 uselib_store='GPGERROR'):
129 conf.env.append_value("CCFLAGS", "-DEVCOM_HAVE_GNUTLS=1")
130 conf.env.append_value("CXXFLAGS", "-DEVCOM_HAVE_GNUTLS=1")
131
Ryan1a126ed2009-04-04 12:50:15132 conf.sub_config('deps/libeio')
133 conf.sub_config('deps/libev')
134
Ryan41d89f62009-07-28 10:29:18135 conf_subproject(conf, 'deps/udns', './configure')
Ryan41d89f62009-07-28 10:29:18136
Ryan1a126ed2009-04-04 12:50:15137 conf.define("HAVE_CONFIG_H", 1)
Ryanc62b1242009-04-22 17:55:08138
Ryan1df6d612009-09-03 13:59:48139 conf.env.append_value("CCFLAGS", "-DX_STACKSIZE=%d" % (1024*64))
Ryan427e3f52009-05-14 11:16:45140
Ryan Dahl2b743aa2009-10-27 11:05:38141 # LFS
142 conf.env.append_value('CCFLAGS', '-D_LARGEFILE_SOURCE')
143 conf.env.append_value('CXXFLAGS', '-D_LARGEFILE_SOURCE')
144 conf.env.append_value('CCFLAGS', '-D_FILE_OFFSET_BITS=64')
145 conf.env.append_value('CXXFLAGS', '-D_FILE_OFFSET_BITS=64')
146
Ryan Dahlf4811832009-11-02 23:21:00147 # platform
148 platform_def = '-DPLATFORM=' + sys.platform
149 conf.env.append_value('CCFLAGS', platform_def)
150 conf.env.append_value('CXXFLAGS', platform_def)
151
Ryan67af9582009-04-18 13:35:42152 # Split off debug variant before adding variant specific defines
Ryan7e1350f2009-04-16 09:37:44153 debug_env = conf.env.copy()
154 conf.set_env_name('debug', debug_env)
Ryan7e1350f2009-04-16 09:37:44155
Ryan67af9582009-04-18 13:35:42156 # Configure debug variant
157 conf.setenv('debug')
158 debug_env.set_variant('debug')
Ryan8ddf9302009-09-02 18:19:52159 debug_env.append_value('CCFLAGS', ['-DDEBUG', '-g', '-O0', '-Wall', '-Wextra'])
160 debug_env.append_value('CXXFLAGS', ['-DDEBUG', '-g', '-O0', '-Wall', '-Wextra'])
Ryan67af9582009-04-18 13:35:42161 conf.write_config_header("config.h")
162
163 # Configure default variant
164 conf.setenv('default')
Ryan8ddf9302009-09-02 18:19:52165 conf.env.append_value('CCFLAGS', ['-DNDEBUG', '-O3'])
166 conf.env.append_value('CXXFLAGS', ['-DNDEBUG', '-O3'])
Ryan67af9582009-04-18 13:35:42167 conf.write_config_header("config.h")
Ryan63a9cd32009-04-15 08:08:28168
Ryan41d89f62009-07-28 10:29:18169def build_udns(bld):
170 default_build_dir = bld.srcnode.abspath(bld.env_of_name("default"))
Ryan1a126ed2009-04-04 12:50:15171
Ryan41d89f62009-07-28 10:29:18172 default_dir = join(default_build_dir, "deps/udns")
173
174 static_lib = bld.env["staticlib_PATTERN"] % "udns"
175
176 rule = 'cd %s && make'
177
178 default = bld.new_task_gen(
179 target= join("deps/udns", static_lib),
180 rule= rule % default_dir,
181 before= "cxx",
182 install_path= None
183 )
184
185 bld.env["CPPPATH_UDNS"] = "deps/udns"
Ryan Dahlfc937aa2009-10-27 21:50:46186 t = join(bld.srcnode.abspath(bld.env_of_name("default")), default.target)
187 bld.env_of_name('default')["LINKFLAGS_UDNS"] = [t]
Ryan41d89f62009-07-28 10:29:18188
189 if bld.env["USE_DEBUG"]:
190 debug_build_dir = bld.srcnode.abspath(bld.env_of_name("debug"))
191 debug_dir = join(debug_build_dir, "deps/udns")
192 debug = default.clone("debug")
193 debug.rule = rule % debug_dir
Ryan Dahlfc937aa2009-10-27 21:50:46194 t = join(bld.srcnode.abspath(bld.env_of_name("debug")), debug.target)
195 bld.env_of_name('debug')["LINKFLAGS_UDNS"] = [t]
Ryan Dahlbf0d2782009-10-03 20:42:03196 bld.install_files('${PREFIX}/include/node/', 'deps/udns/udns.h')
Ryan41d89f62009-07-28 10:29:18197
Ryan Dahl53ebe752009-10-08 21:20:14198def v8_cmd(bld, variant):
199 scons = join(cwd, 'tools/scons/scons.py')
Ryan1a126ed2009-04-04 12:50:15200 deps_src = join(bld.path.abspath(),"deps")
Ryan1a126ed2009-04-04 12:50:15201 v8dir_src = join(deps_src,"v8")
Ryan Dahl53ebe752009-10-08 21:20:14202
Ryan Dahlbc9b3432009-10-02 12:10:40203 # NOTE: We want to compile V8 to export its symbols. I.E. Do not want
204 # -fvisibility=hidden. When using dlopen() it seems that the loaded DSO
205 # cannot see symbols in the executable which are hidden, even if the
206 # executable is statically linked together...
Ryan8ddf9302009-09-02 18:19:52207
Ryan Dahl7d9d8812009-10-26 21:27:52208 # XXX Remove this when v8 defaults x86_64 to native builds
Ryan Dahld85724d2009-10-08 22:34:39209 arch = ""
Ryan Dahl7d9d8812009-10-26 21:27:52210 if bld.env['DEST_CPU'] == 'x86_64':
Ryan Dahld85724d2009-10-08 22:34:39211 arch = "arch=x64"
212
213 if variant == "default":
214 mode = "release"
215 else:
216 mode = "debug"
Ryana4593e32009-04-23 11:18:38217
Ryan Dahl53ebe752009-10-08 21:20:14218 cmd_R = 'python %s -C %s -Y %s visibility=default mode=%s %s library=static snapshot=on'
219
220 cmd = cmd_R % ( scons
221 , bld.srcnode.abspath(bld.env_of_name(variant))
222 , v8dir_src
223 , mode
224 , arch
225 )
226 return cmd
227
228
229def build_v8(bld):
Ryan1a126ed2009-04-04 12:50:15230 v8 = bld.new_task_gen(
Ryan Dahl59b7a1b2009-10-09 10:49:48231 source = 'deps/v8/SConstruct '
232 + bld.path.ant_glob('v8/include/*')
233 + bld.path.ant_glob('v8/src/*'),
Ryan Dahl53ebe752009-10-08 21:20:14234 target = bld.env["staticlib_PATTERN"] % "v8",
235 rule = v8_cmd(bld, "default"),
236 before = "cxx",
237 install_path = None
Ryan1a126ed2009-04-04 12:50:15238 )
Ryan Dahl8b62e862009-10-10 09:58:36239 v8.uselib = "EXECINFO"
Ryan1a126ed2009-04-04 12:50:15240 bld.env["CPPPATH_V8"] = "deps/v8/include"
Ryan Dahlfc937aa2009-10-27 21:50:46241 t = join(bld.srcnode.abspath(bld.env_of_name("default")), v8.target)
242 bld.env_of_name('default')["LINKFLAGS_V8"] = ["-pthread", t]
Ryana4593e32009-04-23 11:18:38243
244 ### v8 debug
Ryan29b528c2009-04-23 15:29:31245 if bld.env["USE_DEBUG"]:
Ryan29b528c2009-04-23 15:29:31246 v8_debug = v8.clone("debug")
Ryan Dahl53ebe752009-10-08 21:20:14247 v8_debug.rule = v8_cmd(bld, "debug")
248 v8_debug.target = bld.env["staticlib_PATTERN"] % "v8_g"
Ryan Dahl8b62e862009-10-10 09:58:36249 v8_debug.uselib = "EXECINFO"
Ryan Dahlfc937aa2009-10-27 21:50:46250 t = join(bld.srcnode.abspath(bld.env_of_name("debug")), v8_debug.target)
251 bld.env_of_name('debug')["LINKFLAGS_V8"] = ["-pthread", t]
Ryan1a126ed2009-04-04 12:50:15252
Ryan Dahl53ebe752009-10-08 21:20:14253 bld.install_files('${PREFIX}/include/node/', 'deps/v8/include/*.h')
Ryan2b6d7242009-06-20 13:07:10254
Ryan41d89f62009-07-28 10:29:18255def build(bld):
256 bld.add_subdirs('deps/libeio deps/libev')
257
258 build_udns(bld)
259 build_v8(bld)
260
Ryan0fb0af32009-07-25 15:52:21261 ### evcom
Ryan Dahl122e74b2009-10-27 21:26:53262 evcom = bld.new_task_gen("cc")
Ryan0fb0af32009-07-25 15:52:21263 evcom.source = "deps/evcom/evcom.c"
264 evcom.includes = "deps/evcom/ deps/libev/"
265 evcom.name = "evcom"
266 evcom.target = "evcom"
Rhys Jonesb6dda612009-11-22 02:58:08267 evcom.uselib = "GPGERROR GNUTLS"
Ryan0fb0af32009-07-25 15:52:21268 evcom.install_path = None
Ryan29b528c2009-04-23 15:29:31269 if bld.env["USE_DEBUG"]:
Ryan0fb0af32009-07-25 15:52:21270 evcom.clone("debug")
Ryan Dahlbf0d2782009-10-03 20:42:03271 bld.install_files('${PREFIX}/include/node/', 'deps/evcom/evcom.h')
Ryan1a126ed2009-04-04 12:50:15272
Ryan5a071ad2009-05-03 12:09:16273 ### http_parser
Ryan Dahl122e74b2009-10-27 21:26:53274 http_parser = bld.new_task_gen("cc")
Ryan5a071ad2009-05-03 12:09:16275 http_parser.source = "deps/http_parser/http_parser.c"
276 http_parser.includes = "deps/http_parser/"
277 http_parser.name = "http_parser"
278 http_parser.target = "http_parser"
279 http_parser.install_path = None
Ryan29b528c2009-04-23 15:29:31280 if bld.env["USE_DEBUG"]:
Ryan5a071ad2009-05-03 12:09:16281 http_parser.clone("debug")
Ryan1a126ed2009-04-04 12:50:15282
Ryan17c6a672009-08-24 18:25:24283 ### coupling
Ryan Dahl122e74b2009-10-27 21:26:53284 coupling = bld.new_task_gen("cc")
Ryan17c6a672009-08-24 18:25:24285 coupling.source = "deps/coupling/coupling.c"
286 coupling.includes = "deps/coupling/"
287 coupling.name = "coupling"
288 coupling.target = "coupling"
289 coupling.install_path = None
290 if bld.env["USE_DEBUG"]:
291 coupling.clone("debug")
292
Ryan63a9cd32009-04-15 08:08:28293 ### src/native.cc
294 def javascript_in_c(task):
295 env = task.env
296 source = map(lambda x: x.srcpath(env), task.inputs)
297 targets = map(lambda x: x.srcpath(env), task.outputs)
298 js2c.JS2C(source, targets)
299
300 native_cc = bld.new_task_gen(
Ryan Dahld737a062009-11-07 13:37:22301 source='src/node.js',
Ryan Dahl53ebe752009-10-08 21:20:14302 target="src/node_natives.h",
Ryan63a9cd32009-04-15 08:08:28303 before="cxx"
304 )
Ryan8e7bbf22009-04-23 17:26:56305 native_cc.install_path = None
Ryan Dahl4bcb01c2009-10-16 20:53:44306
307 # Add the rule /after/ cloning the debug
308 # This is a work around for an error had in python 2.4.3 (I'll paste the
309 # error that was had into the git commit meessage. git-blame to find out
310 # where.)
Ryan29b528c2009-04-23 15:29:31311 if bld.env["USE_DEBUG"]:
Ryan Dahl4bcb01c2009-10-16 20:53:44312 native_cc_debug = native_cc.clone("debug")
313 native_cc_debug.rule = javascript_in_c
314 native_cc.rule = javascript_in_c
Ryan63a9cd32009-04-15 08:08:28315
Ryan2b6d7242009-06-20 13:07:10316 ### node lib
Ryan8152f9c2009-09-01 12:15:29317 node = bld.new_task_gen("cxx", "program")
318 node.name = "node"
319 node.target = "node"
320 node.source = """
Ryan1a126ed2009-04-04 12:50:15321 src/node.cc
Ryan Dahla5df0f62009-10-27 10:46:58322 src/node_child_process.cc
323 src/node_constants.cc
324 src/node_dns.cc
325 src/node_events.cc
326 src/node_file.cc
327 src/node_http.cc
328 src/node_net.cc
329 src/node_signal_handler.cc
Ryan Dahl8d2f9e82009-11-17 13:07:48330 src/node_stat.cc
Ryan17c6a672009-08-24 18:25:24331 src/node_stdio.cc
Ryan Dahla5df0f62009-10-27 10:46:58332 src/node_timer.cc
Ryan1a126ed2009-04-04 12:50:15333 """
Ryan8152f9c2009-09-01 12:15:29334 node.includes = """
Ryan1a126ed2009-04-04 12:50:15335 src/
336 deps/v8/include
337 deps/libev
Ryan41d89f62009-07-28 10:29:18338 deps/udns
Ryan1a126ed2009-04-04 12:50:15339 deps/libeio
Ryan0fb0af32009-07-25 15:52:21340 deps/evcom
Ryan5a071ad2009-05-03 12:09:16341 deps/http_parser
Ryan17c6a672009-08-24 18:25:24342 deps/coupling
Ryan1a126ed2009-04-04 12:50:15343 """
Ryan Dahl122e74b2009-10-27 21:26:53344 node.add_objects = 'ev eio evcom http_parser coupling'
345 node.uselib_local = ''
Rhys Jonesb6dda612009-11-22 02:58:08346 node.uselib = 'UDNS V8 EXECINFO DL GPGERROR GNUTLS'
347
Ryan8152f9c2009-09-01 12:15:29348 node.install_path = '${PREFIX}/lib'
Ryan8e7bbf22009-04-23 17:26:56349 node.install_path = '${PREFIX}/bin'
350 node.chmod = 0755
351
Ryan4d921992009-08-26 23:11:16352 def subflags(program):
Ryan Dahld979ac92009-10-09 13:00:12353 if os.path.exists(join(cwd, ".git")):
Ryan Dahl962e9292009-10-09 14:16:27354 actual_version=cmd_output("git describe").strip()
Ryan Dahld979ac92009-10-09 13:00:12355 else:
356 actual_version=VERSION
357
Ryanb73264d2009-08-27 00:15:11358 x = { 'CCFLAGS' : " ".join(program.env["CCFLAGS"])
359 , 'CPPFLAGS' : " ".join(program.env["CPPFLAGS"])
360 , 'LIBFLAGS' : " ".join(program.env["LIBFLAGS"])
Ryan Dahld979ac92009-10-09 13:00:12361 , 'VERSION' : actual_version
Ryanb73264d2009-08-27 00:15:11362 , 'PREFIX' : program.env["PREFIX"]
Ryan4d921992009-08-26 23:11:16363 }
Ryan Dahlbf0d2782009-10-03 20:42:03364 return x
Ryan4d921992009-08-26 23:11:16365
Ryan4d921992009-08-26 23:11:16366 # process file.pc.in -> file.pc
Ryan Dahld979ac92009-10-09 13:00:12367
Ryanb73264d2009-08-27 00:15:11368 node_version = bld.new_task_gen('subst', before="cxx")
369 node_version.source = 'src/node_version.h.in'
370 node_version.target = 'src/node_version.h'
371 node_version.dict = subflags(node)
Ryana97dce72009-08-31 09:14:34372 node_version.install_path = '${PREFIX}/include/node'
Ryan4d921992009-08-26 23:11:16373
Ryan29b528c2009-04-23 15:29:31374 if bld.env["USE_DEBUG"]:
Ryan2b6d7242009-06-20 13:07:10375 node_g = node.clone("debug")
376 node_g.target = "node_g"
Ryan4d921992009-08-26 23:11:16377
Ryanb73264d2009-08-27 00:15:11378 node_version_g = node_version.clone("debug")
379 node_version_g.dict = subflags(node_g)
Ryana97dce72009-08-31 09:14:34380 node_version_g.install_path = None
Ryan4d921992009-08-26 23:11:16381
Ryana97dce72009-08-31 09:14:34382
383 bld.install_files('${PREFIX}/include/node/', """
384 config.h
385 src/node.h
Ryan Dahl5f466c82009-10-27 19:17:03386 src/node_object_wrap.h
387 src/node_events.h
388 src/node_net.h
Ryan Dahlbf0d2782009-10-03 20:42:03389 """)
390
391 # Only install the man page if it exists.
392 # Do 'make doc install' to build and install it.
393 if os.path.exists('doc/node.1'):
394 bld.install_files('${PREFIX}/share/man/man1/', 'doc/node.1')
395
396 bld.install_files('${PREFIX}/bin/', 'bin/*', chmod=0755)
Ryan Dahl6f17ca52009-10-03 17:08:05397
398 # Why am I using two lines? Because WAF SUCKS.
Ryan Dahlbf0d2782009-10-03 20:42:03399 bld.install_files('${PREFIX}/lib/node/wafadmin', 'tools/wafadmin/*.py')
400 bld.install_files('${PREFIX}/lib/node/wafadmin/Tools', 'tools/wafadmin/Tools/*.py')
Ryan Dahl6f17ca52009-10-03 17:08:05401
Ryan Dahlbf0d2782009-10-03 20:42:03402 bld.install_files('${PREFIX}/lib/node/libraries/', 'lib/*.js')
Ryan Dahl132d6852009-10-27 17:11:07403
404def shutdown():
405 Options.options.debug
406 # HACK to get binding.node out of build directory.
407 # better way to do this?
408 if not Options.commands['clean']:
409 if os.path.exists('build/default/node') and not os.path.exists('node'):
410 os.symlink('build/default/node', 'node')
411 if os.path.exists('build/debug/node_g') and not os.path.exists('node_g'):
412 os.symlink('build/debug/node_g', 'node_g')
413 else:
414 if os.path.exists('node'): os.unlink('node')
415 if os.path.exists('node_g'): os.unlink('node_g')