1 // Copyright (c) 2013 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "tools/gn/functions.h"
7 #include "tools/gn/config_values_generator.h"
8 #include "tools/gn/err.h"
9 #include "tools/gn/parse_tree.h"
10 #include "tools/gn/scope.h"
11 #include "tools/gn/target_generator.h"
12 #include "tools/gn/value.h"
13 #include "tools/gn/variables.h"
15 #define DEPENDENT_CONFIG_VARS \
16 " Dependent configs: all_dependent_configs, direct_dependent_configs\n"
18 " Deps: data, datadeps, deps, forward_dependent_configs_from, hard_dep\n"
19 #define GENERAL_TARGET_VARS \
20 " General: configs, external, source_prereqs, sources\n"
26 Value
ExecuteGenericTarget(const char* target_type
,
28 const FunctionCallNode
* function
,
29 const std::vector
<Value
>& args
,
32 if (!EnsureNotProcessingImport(function
, scope
, err
) ||
33 !EnsureNotProcessingBuildConfig(function
, scope
, err
))
35 Scope
block_scope(scope
);
36 if (!FillTargetBlockScope(scope
, function
, target_type
, block
,
37 args
, &block_scope
, err
))
40 block
->ExecuteBlockInScope(&block_scope
, err
);
44 TargetGenerator::GenerateTarget(&block_scope
, function
->function(), args
,
49 block_scope
.CheckForUnusedVars(err
);
55 // component -------------------------------------------------------------------
57 const char kComponent
[] = "component";
58 const char kComponent_Help
[] =
59 "component: Declare a component target.\n"
61 " A component is a shared library, static library, or source set\n"
62 " depending on the component mode. This allows a project to separate\n"
63 " out a build into shared libraries for faster devlopment (link time is\n"
64 " reduced) but to switch to a static build for releases (for better\n"
67 " To use this function you must set the value of the \"component_mode\n"
68 " variable to one of the following strings:\n"
69 " - \"shared_library\"\n"
70 " - \"static_library\"\n"
72 " It is an error to call \"component\" without defining the mode\n"
73 " (typically this is done in the master build configuration file).\n";
75 Value
RunComponent(Scope
* scope
,
76 const FunctionCallNode
* function
,
77 const std::vector
<Value
>& args
,
80 // A component is either a shared or static library, depending on the value
81 // of |component_mode|.
82 const Value
* component_mode_value
=
83 scope
->GetValue(variables::kComponentMode
);
85 static const char helptext
[] =
86 "You're declaring a component here but have not defined "
87 "\"component_mode\" to\neither \"shared_library\" or \"static_library\".";
88 if (!component_mode_value
) {
89 *err
= Err(function
->function(), "No component mode set.", helptext
);
92 if (component_mode_value
->type() != Value::STRING
||
93 (component_mode_value
->string_value() != functions::kSharedLibrary
&&
94 component_mode_value
->string_value() != functions::kStaticLibrary
&&
95 component_mode_value
->string_value() != functions::kSourceSet
)) {
96 *err
= Err(function
->function(), "Invalid component mode set.", helptext
);
99 const std::string
& component_mode
= component_mode_value
->string_value();
101 if (!EnsureNotProcessingImport(function
, scope
, err
))
103 Scope
block_scope(scope
);
104 if (!FillTargetBlockScope(scope
, function
, component_mode
.c_str(), block
,
105 args
, &block_scope
, err
))
108 block
->ExecuteBlockInScope(&block_scope
, err
);
109 if (err
->has_error())
112 TargetGenerator::GenerateTarget(&block_scope
, function
->function(), args
,
113 component_mode
, err
);
117 // copy ------------------------------------------------------------------------
119 const char kCopy
[] = "copy";
120 const char kCopy_Help
[] =
121 "copy: Declare a target that copies files.\n"
123 "File name handling\n"
125 " All output files must be inside the output directory of the build.\n"
126 " You would generally use |$target_out_dir| or |$target_gen_dir| to\n"
127 " reference the output or generated intermediate file directories,\n"
130 " Both \"sources\" and \"outputs\" must be specified. Sources can\n"
131 " as many files as you want, but there can only be one item in the\n"
132 " outputs list (plural is used for the name for consistency with\n"
133 " other target types).\n"
135 " If there is more than one source file, your output name should specify\n"
136 " a mapping from each source files to output file names using source\n"
137 " expansion (see \"gn help source_expansion\"). The placeholders will\n"
138 " will look like \"{{source_name_part}}\", for example.\n"
142 " # Write a rule that copies a checked-in DLL to the output directory.\n"
143 " copy(\"mydll\") {\n"
144 " sources = [ \"mydll.dll\" ]\n"
145 " outputs = [ \"$target_out_dir/mydll.dll\" ]\n"
148 " # Write a rule to copy several files to the target generated files\n"
150 " copy(\"myfiles\") {\n"
151 " sources = [ \"data1.dat\", \"data2.dat\", \"data3.dat\" ]\n"
153 " # Use source expansion to generate output files with the\n"
154 " # corresponding file names in the gen dir. This will just copy each\n"
156 " outputs = [ \"$target_gen_dir/{{source_file_part}}\" ]\n"
159 Value
RunCopy(const FunctionCallNode
* function
,
160 const std::vector
<Value
>& args
,
163 if (!EnsureNotProcessingImport(function
, scope
, err
) ||
164 !EnsureNotProcessingBuildConfig(function
, scope
, err
))
166 TargetGenerator::GenerateTarget(scope
, function
->function(), args
,
167 functions::kCopy
, err
);
171 // custom ----------------------------------------------------------------------
173 const char kCustom
[] = "custom";
174 const char kCustom_Help
[] =
175 "custom: Declare a script-generated target.\n"
177 " This target type allows you to run a script over a set of source\n"
178 " files and generate a set of output files.\n"
180 " The script will be executed with the given arguments with the current\n"
181 " directory being that of the root build directory. If you pass\n"
182 " to your script, see \"gn help to_build_path\" for how to convert\n"
183 " file names to be relative to the build directory (file names in the\n"
184 " sources, outputs, and source_prereqs will be all treated as relative\n"
185 " to the current build file and converted as needed automatically).\n"
187 " There are two modes. The first mode is the \"per-file\" mode where you\n"
188 " specify a list of sources and the script is run once for each one as a\n"
189 " build rule. In this case, each file specified in the |outputs|\n"
190 " variable must be unique when applied to each source file (normally you\n"
191 " would reference |{{source_name_part}}| from within each one) or the\n"
192 " build system will get confused about how to build those files. You\n"
193 " should use the |source_prereqs| variable to list all additional\n"
194 " dependencies of your script: these will be added as dependencies for\n"
195 " each build step.\n"
197 " The second mode is when you just want to run a script once rather than\n"
198 " as a general rule over a set of files. In this case you don't list any\n"
199 " sources. Dependencies of your script are specified only in the\n"
200 " |source_prereqs| variable and your |outputs| variable should just list\n"
203 "File name handling\n"
205 " All output files must be inside the output directory of the build.\n"
206 " You would generally use |$target_out_dir| or |$target_gen_dir| to\n"
207 " reference the output or generated intermediate file directories,\n"
210 " You can specify a mapping from source files to output files using\n"
211 " source expansion (see \"gn help source_expansion\"). The placeholders\n"
212 " will look like \"{{source}}\", for example, and can appear in\n"
213 " either the outputs or the args lists.\n"
217 " args, deps, outputs, script*, source_prereqs, sources\n"
222 " # Runs the script over each IDL file. The IDL script will generate\n"
223 " # both a .cc and a .h file for each input.\n"
224 " custom(\"general_rule\") {\n"
225 " script = \"idl_processor.py\"\n"
226 " sources = [ \"foo.idl\", \"bar.idl\" ]\n"
227 " source_prereqs = [ \"my_configuration.txt\" ]\n"
228 " outputs = [ \"$target_gen_dir/{{source_name_part}}.h\",\n"
229 " \"$target_gen_dir/{{source_name_part}}.cc\" ]\n"
231 " # Note that since \"args\" is opaque to GN, if you specify paths\n"
232 " # here, you will need to convert it to be relative to the build\n"
233 " # directory using \"to_build_path()\".\n"
234 " args = [ \"{{source}}\",\n"
236 " to_build_path(relative_target_gen_dir) + \"/\" +\n"
237 " {{source_name_part}}.h\" ]\n"
240 " custom(\"just_run_this_guy_once\") {\n"
241 " script = \"doprocessing.py\"\n"
242 " source_prereqs = [ \"my_configuration.txt\" ]\n"
243 " outputs = [ \"$target_gen_dir/insightful_output.txt\" ]\n"
244 " args = [ \"--output_dir\", to_build_path(target_gen_dir) ]\n"
247 Value
RunCustom(Scope
* scope
,
248 const FunctionCallNode
* function
,
249 const std::vector
<Value
>& args
,
252 return ExecuteGenericTarget(functions::kCustom
, scope
, function
, args
,
256 // executable ------------------------------------------------------------------
258 const char kExecutable
[] = "executable";
259 const char kExecutable_Help
[] =
260 "executable: Declare an executable target.\n"
264 CONFIG_VALUES_VARS_HELP
266 DEPENDENT_CONFIG_VARS
269 Value
RunExecutable(Scope
* scope
,
270 const FunctionCallNode
* function
,
271 const std::vector
<Value
>& args
,
274 return ExecuteGenericTarget(functions::kExecutable
, scope
, function
, args
,
278 // group -----------------------------------------------------------------------
280 const char kGroup
[] = "group";
281 const char kGroup_Help
[] =
282 "group: Declare a named group of targets.\n"
284 " This target type allows you to create meta-targets that just collect a\n"
285 " set of dependencies into one named target. Groups can additionally\n"
286 " specify configs that apply to their dependents.\n"
288 " Depending on a group is exactly like depending directly on that\n"
289 " group's deps. Direct dependent configs will get automatically fowarded\n"
290 " through the group so you shouldn't need to use\n"
291 " \"forward_dependent_configs_from.\n"
296 DEPENDENT_CONFIG_VARS
297 " Other variables: external\n"
301 " group(\"all\") {\n"
303 " \"//project:runner\",\n"
304 " \"//project:unit_tests\",\n"
308 Value
RunGroup(Scope
* scope
,
309 const FunctionCallNode
* function
,
310 const std::vector
<Value
>& args
,
313 return ExecuteGenericTarget(functions::kGroup
, scope
, function
, args
,
317 // shared_library --------------------------------------------------------------
319 const char kSharedLibrary
[] = "shared_library";
320 const char kSharedLibrary_Help
[] =
321 "shared_library: Declare a shared library target.\n"
323 " A shared library will be specified on the linker line for targets\n"
324 " listing the shared library in its \"deps\". If you don't want this\n"
325 " (say you dynamically load the library at runtime), then you should\n"
326 " depend on the shared library via \"datadeps\" instead.\n"
330 CONFIG_VALUES_VARS_HELP
332 DEPENDENT_CONFIG_VARS
335 Value
RunSharedLibrary(Scope
* scope
,
336 const FunctionCallNode
* function
,
337 const std::vector
<Value
>& args
,
340 return ExecuteGenericTarget(functions::kSharedLibrary
, scope
, function
, args
,
344 // source_set ------------------------------------------------------------------
346 extern const char kSourceSet
[] = "source_set";
347 extern const char kSourceSet_Help
[] =
348 "source_set: Declare a source set target.\n"
350 " A source set is a collection of sources that get compiled, but are not\n"
351 " linked to produce any kind of library. Instead, the resulting object\n"
352 " files are implicitly added to the linker line of all targets that\n"
353 " depend on the source set.\n"
355 " In most cases, a source set will behave like a static library, except\n"
356 " no actual library file will be produced. This will make the build go\n"
357 " a little faster by skipping creation of a large static library, while\n"
358 " maintaining the organizational benefits of focused build targets.\n"
360 " The main difference between a source set and a static library is\n"
361 " around handling of exported symbols. Most linkers assume declaring\n"
362 " a function exported means exported from the static library. The linker\n"
363 " can then do dead code elimination to delete code not reachable from\n"
364 " exported functions.\n"
366 " A source set will not do this code elimination since there is no link\n"
367 " step. This allows you to link many sources sets into a shared library\n"
368 " and have the \"exported symbol\" notation indicate \"export from the\n"
369 " final shared library and not from the intermediate targets.\" There is\n"
370 " no way to express this concept when linking multiple static libraries\n"
371 " into a shared library.\n"
375 CONFIG_VALUES_VARS_HELP
377 DEPENDENT_CONFIG_VARS
380 Value
RunSourceSet(Scope
* scope
,
381 const FunctionCallNode
* function
,
382 const std::vector
<Value
>& args
,
385 return ExecuteGenericTarget(functions::kSourceSet
, scope
, function
, args
,
389 // static_library --------------------------------------------------------------
391 const char kStaticLibrary
[] = "static_library";
392 const char kStaticLibrary_Help
[] =
393 "static_library: Declare a static library target.\n"
395 " Make a \".a\" / \".lib\" file.\n"
397 " If you only need the static library for intermediate results in the\n"
398 " build, you should consider a source_set instead since it will skip\n"
399 " the (potentially slow) step of creating the intermediate library file.\n"
403 CONFIG_VALUES_VARS_HELP
405 DEPENDENT_CONFIG_VARS
408 Value
RunStaticLibrary(Scope
* scope
,
409 const FunctionCallNode
* function
,
410 const std::vector
<Value
>& args
,
413 return ExecuteGenericTarget(functions::kStaticLibrary
, scope
, function
, args
,
417 // test ------------------------------------------------------------------------
419 const char kTest
[] = "test";
420 const char kTest_Help
[] =
421 "test: Declares a test target.\n"
423 " This is like an executable target, but is named differently to make\n"
424 " the purpose of the target more obvious. It's possible in the future\n"
425 " we can do some enhancements like \"list all of the tests in a given\n"
428 " See \"gn help executable\" for usage.\n";
430 Value
RunTest(Scope
* scope
,
431 const FunctionCallNode
* function
,
432 const std::vector
<Value
>& args
,
435 return ExecuteGenericTarget(functions::kExecutable
, scope
, function
, args
,
439 } // namespace functions