blob: 7c5abf204f04ff4c6afbade1bbe403ae50552864 [file] [log] [blame]
Armando Montanez70095662020-01-09 14:25:04 -08001# Copyright 2020 The Pigweed Authors
2#
3# Licensed under the Apache License, Version 2.0 (the "License"); you may not
4# use this file except in compliance with the License. You may obtain a copy of
5# the License at
6#
7# https://www.apache.org/licenses/LICENSE-2.0
8#
9# Unless required by applicable law or agreed to in writing, software
10# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
11# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
12# License for the specific language governing permissions and limitations under
13# the License.
14
15import("$dir_pw_build/exec.gni")
16
17# Preprocess a linker script and turn it into a target.
18#
19# Note: to use this template, pw_cc_command must be specified in your target's
20# config. It should match the name of the C compiler your target uses (e.g.
21# arm-none-eabi-gcc).
22#
23# In lieu of direct GN support for linker scripts, this template makes it
24# possible to run the C Preprocessor on a linker script file so defines can
25# be properly evaluated before the linker script is passed to the dir_pw_build
26#
27# TODO(pwbug/53): This template serves as a stand-in until native GN support for
28# linker scripts is added.
29#
30# Args:
31# linker_script: The linker script to send through the C preprocessor.
32#
33# defines: Preprocessor defines to apply when running the C preprocessor.
34#
35# cflags: Flags to pass to the C compiler.
36#
37# inputs: Files that, when changed, should trigger a re-build of the linker
38# script. linker_script is implicitly added to this by the template.
39#
40# Example:
41#
42# pw_linker_script("generic_linker_script") {
43# defines = [
44# "PW_HEAP_SIZE=1K",
45# "PW_NOINIT_SIZE=512"
46# ]
47# linker_script = "basic_script.ld"
48# }
49#
50template("pw_linker_script") {
51 assert(
52 defined(pw_cc_command) && pw_cc_command != "",
53 "pw_cc_command has not been properly configured. This variable must be " +
54 "defined to enable linker script preprocessing.")
55
56 assert(
57 defined(invoker.linker_script) && invoker.linker_script != "",
58 "$target_name did not set `linker_script` to refer to a valid linker " +
59 "script. This variable is required for linker script targets.")
60
61 _final_linker_script = "${target_gen_dir}/${target_name}_final.ld"
62
63 # This action invokes the C compiler provided by the target to preprocess the
64 # linker script.
65 pw_exec("${target_name}_preprocess") {
66 program = pw_cc_command
Rob Mohra0ba54f2020-02-27 11:43:49 -080067 inputs = [ invoker.linker_script ]
Armando Montanez70095662020-01-09 14:25:04 -080068 args = [
69 # Run compiler in preprocessor-only mode.
70 "-E",
71
72 # Do not generate linemarkers in output.
73 "-P",
74
75 # Do not discard comments.
76 "-C",
77
78 # Treat the following file as a C file.
79 "-x",
80 "c",
Alexei Frolovee945602020-01-23 12:51:49 -080081 get_path_info(invoker.linker_script, "abspath"),
Armando Montanez70095662020-01-09 14:25:04 -080082 ]
83
84 # Include any explicitly listed c flags.
85 if (defined(invoker.cflags)) {
86 args += cflags
87 }
88
89 # Add defines.
90 if (defined(invoker.defines)) {
91 args += process_file_template(invoker.defines, "-D{{source_name_part}}")
92 }
93
94 # Set output file.
95 args += [
96 "-o",
97 _final_linker_script,
98 ]
Rob Mohra0ba54f2020-02-27 11:43:49 -080099 outputs = [ _final_linker_script ]
Armando Montanez70095662020-01-09 14:25:04 -0800100 }
101
102 # This config adds a the linker script produced by the preprocess action to
103 # the linker flags.
104 config("${target_name}_config") {
Rob Mohra0ba54f2020-02-27 11:43:49 -0800105 inputs = [ invoker.linker_script ]
Armando Montanez70095662020-01-09 14:25:04 -0800106 if (!defined(invoker.ldflags)) {
107 ldflags = []
108 }
109 ldflags += [ "-T" + rebase_path(_final_linker_script) ]
110 }
111
112 # The target that adds the linker script config to this library and everything
113 # that depends on it.
114 source_set(target_name) {
Rob Mohra0ba54f2020-02-27 11:43:49 -0800115 inputs = [ _final_linker_script ]
Armando Montanez70095662020-01-09 14:25:04 -0800116 if (defined(invoker.inputs)) {
117 inputs += invoker.inputs
118 }
119 all_dependent_configs = [ ":${target_name}_config" ]
Rob Mohra0ba54f2020-02-27 11:43:49 -0800120 deps = [ ":${target_name}_preprocess" ]
Armando Montanez70095662020-01-09 14:25:04 -0800121 }
122}