Tizen 2.0 Release
[profile/ivi/osmesa.git] / scons / custom.py
1 """custom
2
3 Custom builders and methods.
4
5 """
6
7 #
8 # Copyright 2008 Tungsten Graphics, Inc., Cedar Park, Texas.
9 # All Rights Reserved.
10 #
11 # Permission is hereby granted, free of charge, to any person obtaining a
12 # copy of this software and associated documentation files (the
13 # "Software"), to deal in the Software without restriction, including
14 # without limitation the rights to use, copy, modify, merge, publish,
15 # distribute, sub license, and/or sell copies of the Software, and to
16 # permit persons to whom the Software is furnished to do so, subject to
17 # the following conditions:
18 #
19 # The above copyright notice and this permission notice (including the
20 # next paragraph) shall be included in all copies or substantial portions
21 # of the Software.
22 #
23 # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
24 # OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
25 # MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT.
26 # IN NO EVENT SHALL TUNGSTEN GRAPHICS AND/OR ITS SUPPLIERS BE LIABLE FOR
27 # ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
28 # TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
29 # SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
30 #
31
32
33 import os
34 import os.path
35 import re
36
37 import SCons.Action
38 import SCons.Builder
39 import SCons.Scanner
40
41 import fixes
42
43
44 def quietCommandLines(env):
45     # Quiet command lines
46     # See also http://www.scons.org/wiki/HidingCommandLinesInOutput
47     env['ASCOMSTR'] = "  Assembling $SOURCE ..."
48     env['ASPPCOMSTR'] = "  Assembling $SOURCE ..."
49     env['CCCOMSTR'] = "  Compiling $SOURCE ..."
50     env['SHCCCOMSTR'] = "  Compiling $SOURCE ..."
51     env['CXXCOMSTR'] = "  Compiling $SOURCE ..."
52     env['SHCXXCOMSTR'] = "  Compiling $SOURCE ..."
53     env['ARCOMSTR'] = "  Archiving $TARGET ..."
54     env['RANLIBCOMSTR'] = "  Indexing $TARGET ..."
55     env['LINKCOMSTR'] = "  Linking $TARGET ..."
56     env['SHLINKCOMSTR'] = "  Linking $TARGET ..."
57     env['LDMODULECOMSTR'] = "  Linking $TARGET ..."
58     env['SWIGCOMSTR'] = "  Generating $TARGET ..."
59     env['LEXCOMSTR'] = "  Generating $TARGET ..."
60     env['YACCCOMSTR'] = "  Generating $TARGET ..."
61     env['CODEGENCOMSTR'] = "  Generating $TARGET ..."
62     env['INSTALLSTR'] = "  Installing $TARGET ..."
63
64
65 def createConvenienceLibBuilder(env):
66     """This is a utility function that creates the ConvenienceLibrary
67     Builder in an Environment if it is not there already.
68
69     If it is already there, we return the existing one.
70
71     Based on the stock StaticLibrary and SharedLibrary builders.
72     """
73
74     try:
75         convenience_lib = env['BUILDERS']['ConvenienceLibrary']
76     except KeyError:
77         action_list = [ SCons.Action.Action("$ARCOM", "$ARCOMSTR") ]
78         if env.Detect('ranlib'):
79             ranlib_action = SCons.Action.Action("$RANLIBCOM", "$RANLIBCOMSTR")
80             action_list.append(ranlib_action)
81
82         convenience_lib = SCons.Builder.Builder(action = action_list,
83                                   emitter = '$LIBEMITTER',
84                                   prefix = '$LIBPREFIX',
85                                   suffix = '$LIBSUFFIX',
86                                   src_suffix = '$SHOBJSUFFIX',
87                                   src_builder = 'SharedObject')
88         env['BUILDERS']['ConvenienceLibrary'] = convenience_lib
89
90     return convenience_lib
91
92
93 # TODO: handle import statements with multiple modules
94 # TODO: handle from import statements
95 import_re = re.compile(r'^import\s+(\S+)$', re.M)
96
97 def python_scan(node, env, path):
98     # http://www.scons.org/doc/0.98.5/HTML/scons-user/c2781.html#AEN2789
99     contents = node.get_contents()
100     source_dir = node.get_dir()
101     imports = import_re.findall(contents)
102     results = []
103     for imp in imports:
104         for dir in path:
105             file = os.path.join(str(dir), imp.replace('.', os.sep) + '.py')
106             if os.path.exists(file):
107                 results.append(env.File(file))
108                 break
109             file = os.path.join(str(dir), imp.replace('.', os.sep), '__init__.py')
110             if os.path.exists(file):
111                 results.append(env.File(file))
112                 break
113     return results
114
115 python_scanner = SCons.Scanner.Scanner(function = python_scan, skeys = ['.py'])
116
117
118 def code_generate(env, script, target, source, command):
119     """Method to simplify code generation via python scripts.
120
121     http://www.scons.org/wiki/UsingCodeGenerators
122     http://www.scons.org/doc/0.98.5/HTML/scons-user/c2768.html
123     """
124
125     # We're generating code using Python scripts, so we have to be
126     # careful with our scons elements.  This entry represents
127     # the generator file *in the source directory*.
128     script_src = env.File(script).srcnode()
129
130     # This command creates generated code *in the build directory*.
131     command = command.replace('$SCRIPT', script_src.path)
132     action = SCons.Action.Action(command, "$CODEGENCOMSTR")
133     code = env.Command(target, source, action)
134
135     # Explicitly mark that the generated code depends on the generator,
136     # and on implicitly imported python modules
137     path = (script_src.get_dir(),)
138     deps = [script_src]
139     deps += script_src.get_implicit_deps(env, python_scanner, path)
140     env.Depends(code, deps)
141
142     # Running the Python script causes .pyc files to be generated in the
143     # source directory.  When we clean up, they should go too. So add side
144     # effects for .pyc files
145     for dep in deps:
146         pyc = env.File(str(dep) + 'c')
147         env.SideEffect(pyc, code)
148
149     return code
150
151
152 def createCodeGenerateMethod(env):
153     env.Append(SCANNERS = python_scanner)
154     env.AddMethod(code_generate, 'CodeGenerate')
155
156
157 def generate(env):
158     """Common environment generation code"""
159
160     verbose = env.get('verbose', False) or not env.get('quiet', True)
161     if not verbose:
162         quietCommandLines(env)
163
164     # Custom builders and methods
165     createConvenienceLibBuilder(env)
166     createCodeGenerateMethod(env)
167
168     # for debugging
169     #print env.Dump()
170
171
172 def exists(env):
173     return 1