]>
Commit | Line | Data |
---|---|---|
cd1a2927 MT |
1 | ############################################################################### |
2 | # LFSMake # | |
3 | # by Rod Roark <rod@sunsetsystems.com> # | |
4 | # # | |
5 | # Copyright (C) 2002 Rod Roark # | |
6 | # # | |
7 | # See http://www.lfsmake.org/ for the most current standard version. # | |
8 | # # | |
9 | # These Makefiles are made available under the terms of the Artistic License, # | |
10 | # found at http://www.opensource.org/licenses/artistic-license.html. # | |
11 | ############################################################################### | |
12 | ||
70df8302 MT |
13 | ############################################################################### |
14 | # # | |
15 | # IPFire.org - A linux based firewall # | |
aafdd71b | 16 | # Copyright (C) 2007-2021 IPFire Team <info@ipfire.org> # |
70df8302 MT |
17 | # # |
18 | # This program is free software: you can redistribute it and/or modify # | |
19 | # it under the terms of the GNU General Public License as published by # | |
20 | # the Free Software Foundation, either version 3 of the License, or # | |
21 | # (at your option) any later version. # | |
22 | # # | |
23 | # This program is distributed in the hope that it will be useful, # | |
24 | # but WITHOUT ANY WARRANTY; without even the implied warranty of # | |
25 | # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # | |
26 | # GNU General Public License for more details. # | |
27 | # # | |
28 | # You should have received a copy of the GNU General Public License # | |
29 | # along with this program. If not, see <http://www.gnu.org/licenses/>. # | |
30 | # # | |
31 | ############################################################################### | |
32 | ||
dc7d6b20 | 33 | # Cleanup environment from any variables |
ebaecf2c | 34 | unexport BUILD_ARCH BUILD_PLATFORM BUILDTARGET CROSSTARGET TOOLCHAIN TOOLS_DIR |
0bc5b1de | 35 | unexport XZ_OPT |
dc7d6b20 | 36 | |
4f2a9813 MT |
37 | # Basic Variables |
38 | EMPTY := | |
39 | COMMA := , | |
40 | SPACE := $(EMPTY) $(EMPTY) | |
41 | define NEWLINE | |
42 | ||
43 | ||
44 | endef | |
45 | ||
46 | # Basic Functions | |
47 | join-with = $(subst $(SPACE),$(1),$(strip $(2))) | |
48 | ||
7691a1bf | 49 | PARALLELISM = $(shell echo $$( \ |
99d987bd AF |
50 | if [ -n "$(MAX_PARALLELISM)" ] && [ $(MAX_PARALLELISM) -lt 1 ]; then \ |
51 | echo 1 ; \ | |
52 | elif [ -n "$(MAX_PARALLELISM)" ] && [ $(MAX_PARALLELISM) -lt $(DEFAULT_PARALLELISM) ]; then \ | |
7691a1bf MT |
53 | echo $(MAX_PARALLELISM); \ |
54 | else \ | |
55 | echo $(DEFAULT_PARALLELISM); \ | |
56 | fi) \ | |
57 | ) | |
58 | ||
d53537ce | 59 | MAKETUNING = -j$(PARALLELISM) |
77c863a2 | 60 | |
b7093f17 MT |
61 | ifeq "$(BUILD_ARCH)" "aarch64" |
62 | IS_64BIT = 1 | |
63 | endif | |
64 | ||
b26071d3 MT |
65 | ifeq "$(BUILD_ARCH)" "armv7hl" |
66 | IS_32BIT = 1 | |
67 | endif | |
68 | ||
aafdd71b | 69 | ifeq "$(BUILD_ARCH)" "armv6l" |
b26071d3 MT |
70 | IS_32BIT = 1 |
71 | endif | |
72 | ||
b7093f17 MT |
73 | ifeq "$(BUILD_ARCH)" "x86_64" |
74 | IS_64BIT = 1 | |
75 | endif | |
76 | ||
000ff407 MT |
77 | ifeq "$(BUILD_ARCH)" "riscv64" |
78 | IS_64BIT = 1 | |
79 | endif | |
80 | ||
c0878e43 MT |
81 | ifeq "$(TOOLCHAIN)" "1" |
82 | PREFIX = $(TOOLS_DIR) | |
83 | else | |
84 | PREFIX = /usr | |
85 | endif | |
86 | ||
b37678e9 MT |
87 | TAR_OPTIONS = \ |
88 | --format=pax \ | |
89 | --acls \ | |
90 | --xattrs --xattrs-include='*' | |
27267889 | 91 | |
5621b0ef MT |
92 | ZSTD_OPTIONS = \ |
93 | -T$(PARALLELISM) \ | |
94 | -19 \ | |
95 | --long | |
96 | ||
cd1a2927 MT |
97 | # URLs that are common sources of downloads. If you're having trouble with |
98 | # a site you should change its URL to that of a suitable mirror site. | |
99 | # | |
c484679b MT |
100 | URL_IPFIRE = https://source.ipfire.org/source-2.x |
101 | URL_TOOLCHAIN = https://source.ipfire.org/toolchains | |
8a148bbc | 102 | URL_SOURCE = git.ipfire.org:/pub/sources/source-2.x |
cd1a2927 | 103 | |
cd1a2927 MT |
104 | # Don't change this; it will be overridden by other makefiles where necessary. |
105 | # | |
106 | ROOT = | |
107 | ||
108 | # For most packages tarballs are unpacked here and then deleted after | |
109 | # installation. | |
110 | # | |
111 | DIR_SRC = $(ROOT)/usr/src | |
112 | ||
113 | # Files are downloaded into DIR_TMP and then moved to DIR_DL, to avoid | |
114 | # messes with partially retrieved files. DIR_DL is where we will | |
115 | # save all the files that are downloaded. DIR_INFO contains the | |
116 | # file lists of installed packages. | |
117 | # | |
118 | DIR_DL = $(LFS_BASEDIR)/cache | |
119 | DIR_CHK = $(LFS_BASEDIR)/cache/check | |
3796e3d5 | 120 | DIR_CONF = $(LFS_BASEDIR)/config |
ebaecf2c | 121 | DIR_INFO = $(LFS_BASEDIR)/log |
05207d69 | 122 | DIR_TMP = /tmp |
258bc2e3 | 123 | DIR_TMP_PAK = $(DIR_TMP)/package-$(PROG) |
cd1a2927 | 124 | |
1d9be34e MT |
125 | # Add the compiler location and version and specs to the ccache hash |
126 | CCACHE_COMPILERCHECK += $(shell gcc -dumpspecs 2>/dev/null | md5sum | cut -d ' ' -f1) | |
127 | ||
582aba05 | 128 | # We support EFI on x86_64 and aarch64 |
ba3cbb0c | 129 | ifeq "$(BUILD_ARCH)" "x86_64" |
72d40cfd | 130 | EFI = 1 |
5f387a1b | 131 | EFI_ARCH = x64 |
a1eb7761 | 132 | GRUB_ARCH = $(BUILD_ARCH) |
72d40cfd MT |
133 | endif |
134 | ||
582aba05 MT |
135 | ifeq "$(BUILD_ARCH)" "aarch64" |
136 | EFI = 1 | |
137 | EFI_ARCH = aa64 | |
a1eb7761 | 138 | GRUB_ARCH = arm64 |
582aba05 MT |
139 | endif |
140 | ||
cde41c2e | 141 | # Go |
b4863d7d MT |
142 | export GOARCH |
143 | export GOOS = linux | |
cde41c2e MT |
144 | export GOPATH = $(HOME)/gopath |
145 | ||
b4863d7d MT |
146 | ifeq "$(BUILD_ARCH)" "x86_64" |
147 | GOARCH = amd64 | |
148 | endif | |
149 | ||
150 | ifeq "$(BUILD_ARCH)" "aarch64" | |
151 | GOARCH = arm64 | |
152 | endif | |
153 | ||
112441db | 154 | # Rust |
f50da967 MT |
155 | RUST_ARCH = $(BUILD_ARCH) |
156 | ||
157 | ifeq "$(BUILD_ARCH)" "armv6l" | |
158 | RUST_PLATFORM = arm-unknown-linux-gnueabi | |
159 | else | |
160 | RUST_PLATFORM = $(RUST_ARCH)-unknown-linux-gnu | |
161 | endif | |
162 | ||
4f2a9813 MT |
163 | CARGO_PATH = $(DIR_APP)/.cargo |
164 | CARGO_REGISTRY = /usr/share/cargo/registry | |
165 | ||
166 | CRATE_NAME = $(patsubst rust-%,%,$(firstword $(MAKEFILE_LIST))) | |
167 | CRATE_VER = $(VER) | |
168 | CRATE_PATH = $(CARGO_REGISTRY)/$(CRATE_NAME)-$(CRATE_VER) | |
169 | ||
170 | define CARGO_CONFIG | |
171 | [build] | |
172 | rustflags = [$(call join-with,$(COMMA)$(SPACE),$(foreach flag,$(RUSTFLAGS),"$(flag)"))] | |
173 | ||
174 | [env] | |
175 | CFLAGS = "$(CFLAGS)" | |
176 | CXXFLAGS = "$(CXXFLAGS)" | |
177 | LDFLAGS = "$(LDFLAGS)" | |
178 | ||
179 | [term] | |
180 | verbose = true | |
181 | ||
182 | [source] | |
183 | ||
184 | [source.local-registry] | |
185 | directory = "$(CARGO_REGISTRY)" | |
186 | ||
187 | [source.crates-io] | |
188 | registry = "https://crates.io" | |
189 | replace-with = "local-registry" | |
190 | endef | |
191 | export CARGO_CONFIG | |
192 | ||
c16dd3aa MT |
193 | # Set to false if you want to skip the binary install step |
194 | CARGO_HAS_BIN = true | |
195 | ||
4f2a9813 MT |
196 | CARGO = \ |
197 | CARGOPATH=$(CARGO_PATH) \ | |
1aab9dfb MT |
198 | RUSTC_BOOTSTRAP=1 \ |
199 | cargo \ | |
200 | --offline | |
201 | ||
202 | CARGO_OPTIONS = \ | |
203 | $(MAKETUNING) \ | |
204 | -Z avoid-dev-deps | |
4f2a9813 MT |
205 | |
206 | define CARGO_PREPARE | |
207 | mkdir -p $(CARGO_PATH) && \ | |
208 | echo "$${CARGO_CONFIG}" > $(CARGO_PATH)/config && \ | |
209 | rm -f Cargo.lock | |
210 | endef | |
211 | ||
212 | CARGO_BUILD = \ | |
213 | $(CARGO) \ | |
214 | build \ | |
1aab9dfb MT |
215 | --release \ |
216 | $(CARGO_OPTIONS) | |
4f2a9813 MT |
217 | |
218 | # Checks whether this crate has a right taregt | |
1aab9dfb | 219 | CARGO_TARGET_CHECK = $(CARGO) metadata --format-version 1 --no-deps | \ |
4f2a9813 MT |
220 | jq -e ".packages[].targets[].kind | any(. == \"$(1)\")" | grep -q "true" |
221 | ||
222 | define CARGO_INSTALL | |
223 | mkdir -pv "$(CRATE_PATH)" && \ | |
4f6d6c41 | 224 | if $(call CARGO_TARGET_CHECK,lib) || $(call CARGO_TARGET_CHECK,rlib) || $(call CARGO_TARGET_CHECK,proc-macro); then \ |
4f2a9813 MT |
225 | awk \ |
226 | '/^\\\[((.+\\\.)?((dev|build)-)?dependencies|features)/{f=1;next} /^\\\[/{f=0}; !f' \ | |
227 | < Cargo.toml > Cargo.toml.deps && \ | |
228 | $(CARGO) package -l | grep -wEv "Cargo.(lock|toml.orig)" \ | |
942b7d0d | 229 | | xargs -d "\n" cp -v --parents -a -t $(CRATE_PATH) && \ |
4f2a9813 MT |
230 | install -v -m 644 Cargo.toml.deps $(CRATE_PATH)/Cargo.toml && \ |
231 | echo "{\"files\":{},\"package\":\"\"}" > $(CRATE_PATH)/.cargo-checksum.json; \ | |
232 | fi && \ | |
c16dd3aa | 233 | if $(CARGO_HAS_BIN) && $(call CARGO_TARGET_CHECK,bin); then \ |
1aab9dfb | 234 | $(CARGO) install $(CARGO_OPTIONS) --no-track --path .; \ |
4f2a9813 MT |
235 | fi |
236 | endef | |
112441db | 237 | |
cd1a2927 MT |
238 | ############################################################################### |
239 | # Common Macro Definitions | |
240 | ############################################################################### | |
241 | ||
242 | # For each package we create a list of files that it installed under | |
ebaecf2c | 243 | # log/<TARGET> name. Modified files are not identified |
cd1a2927 MT |
244 | # |
245 | define FIND_FILES | |
246 | cd $(ROOT)/ && find -mount \ | |
6c4cc7ea | 247 | \( -path '.$(TOOLS_DIR)' -or -path './tmp' -or -path './usr/src' \ |
1b169a72 | 248 | -or -path './run' -or -path './dev' -or -path './proc' \ |
6ede1975 | 249 | -or -path './install' -or -path '.*/__pycache__' \) -prune -or -print | sort |
cd1a2927 MT |
250 | endef |
251 | ||
252 | # This is common starting logic for builds. | |
253 | # | |
254 | ifeq "$(ROOT)" "" | |
255 | define PREBUILD | |
256 | echo "====================================== Installing $(THISAPP) ..." | |
257 | @echo "Install started; saving file list to $(DIR_SRC)/lsalr ..." | |
258 | @if [ ! -f $(DIR_SRC)/lsalr ]; then $(FIND_FILES) > $(DIR_SRC)/lsalr; fi | |
03ad5f93 MT |
259 | # Fix installation on partial rebuild, so modules install where they should |
260 | # and not everytime on the last compiled kernel | |
261 | if [ -f $(DIR_SRC)/linux-$(KVER) ]; then \ | |
fdf0c7c1 AF |
262 | cd $(DIR_SRC)/linux-$(KVER) && \ |
263 | sed -i -e 's+^EXTRAVERSION.*$$+EXTRAVERSION\ =\ $(word 4,$(subst ., .,$(KVER)))-ipfire$(KCFG)+' Makefile; \ | |
03ad5f93 | 264 | fi |
cd1a2927 MT |
265 | endef |
266 | else | |
267 | define PREBUILD | |
268 | echo "====================================== Installing $(THISAPP) ..." | |
269 | endef | |
270 | endif | |
271 | ||
272 | # Common end-of-installation logic for Stage 2 and beyond. | |
273 | # | |
274 | ifeq "$(ROOT)" "" | |
275 | define POSTBUILD | |
f5ad8f59 MT |
276 | @echo "Updating linker cache..." |
277 | @type -p ldconfig >/dev/null && ldconfig || : | |
03ad5f93 | 278 | @echo "Install done; saving file list to $(TARGET) ..." |
06fd5cdd | 279 | @rm -rf $(GOPATH) /root/.cargo |
cd1a2927 | 280 | @$(FIND_FILES) > $(DIR_SRC)/lsalrnew |
e21bd39e | 281 | @diff $(DIR_SRC)/lsalr $(DIR_SRC)/lsalrnew | grep '^> ' | sed 's/^> //' > $(TARGET)_diff |
cd1a2927 MT |
282 | @cp -f $(DIR_SRC)/lsalrnew $(DIR_SRC)/lsalr |
283 | @rm -f $(DIR_SRC)/lsalrnew | |
0145f271 | 284 | sed -i -e 's+.\/++' $(TARGET)_diff |
cd1a2927 MT |
285 | # compare roofile ( same name as lfs script) with the list of installed files |
286 | # special cases | |
03ad5f93 | 287 | # - if the corresponding rootfile is not found, touch $(TARGET)_missing_rootfile |
0145f271 | 288 | # - on a partial rebuild without a new file inside TARGET_diff, just touch TARGET |
03ad5f93 MT |
289 | # $(TARGET)_diff : result of the diff |
290 | # ROOTFILE : reference of include/exclude files | |
291 | # $(TARGET)_rootfile : ROOTFILE with KVER replacement | |
292 | # $(TARGET) : log result with {commented|include|added} files | |
0145f271 | 293 | if [ -s "$(TARGET)_diff" ]; then \ |
fdf0c7c1 | 294 | LFS_SCRIPT=$(firstword $(MAKEFILE_LIST))$(KCFG); \ |
03ad5f93 | 295 | echo $(LFS_SCRIPT); \ |
a4923c7f | 296 | ROOTFILE=$$(find -L $(DIR_SRC)/config/rootfiles/{common,packages}/{$(BUILD_ARCH),} -maxdepth 1 -type f -name $$LFS_SCRIPT 2>/dev/null | head -1); \ |
03ad5f93 MT |
297 | if [ "$$ROOTFILE" = "" ]; then \ |
298 | touch $(TARGET)_missing_rootfile; \ | |
299 | ROOTFILE=$(TARGET)_missing_rootfile ; \ | |
300 | echo "error $$LFS_SCRIPT not found in config/rootfiles"; \ | |
301 | fi; \ | |
289a86a3 | 302 | sed -e "s/BUILDTARGET/$(BUILDTARGET)/g" -e "s/KVER/$(KVER)/g" -e "s/xxxMACHINExxx/$(BUILD_ARCH)/g" $$ROOTFILE > $(TARGET)_rootfile; \ |
0145f271 | 303 | for line in `cat $(TARGET)_diff`; do \ |
c9444dcf AF |
304 | if grep -qG "^#$$line$$" $(TARGET)_rootfile; then echo "#$$line" >> $(TARGET); \ |
305 | elif grep -qG "^$$line$$" $(TARGET)_rootfile ; then echo "$$line" >> $(TARGET); \ | |
cd1a2927 MT |
306 | else echo "+$$line" >> $(TARGET); \ |
307 | fi; \ | |
308 | done; \ | |
0f3c1311 AF |
309 | for line in `grep -v "^#" $(TARGET)_rootfile`; do \ |
310 | if ! grep -qG "^$$line$$" $(TARGET)_diff ; then echo "-$$line" >> $(TARGET); \ | |
311 | fi; \ | |
312 | done; \ | |
0145f271 | 313 | rm -f $(TARGET)_rootfile; \ |
cd1a2927 MT |
314 | else \ |
315 | touch $(TARGET); \ | |
316 | fi | |
0145f271 | 317 | @rm -f $(TARGET)_diff |
cd1a2927 MT |
318 | endef |
319 | else | |
320 | define POSTBUILD | |
321 | @echo "===================================== Install done for $(THISAPP)." | |
322 | touch $(TARGET) | |
323 | endef | |
324 | endif | |
325 | ||
326 | define CHECK | |
327 | @echo -e "$(MESSAGE)Check: $($(notdir $@))" | |
319d3289 | 328 | wget -T 120 -t 1 --spider -nv -U "IPFireSourceGrabber/2.x" $($(notdir $@)) -O /dev/null |
cd1a2927 MT |
329 | @touch $(DIR_CHK)/$(notdir $@) |
330 | endef | |
331 | ||
332 | define LOAD | |
333 | @echo -e "$(MESSAGE)Download: $($(notdir $@))" | |
e99d32db | 334 | wget -T 60 -t 1 -nv -U "IPFireSourceGrabber/2.x" $($(notdir $@)) -O $(DIR_TMP)/$(notdir $@) |
cd1a2927 MT |
335 | [ "$($(notdir $@)_MD5)" = `md5sum $(DIR_TMP)/$(notdir $@) | awk '{ print $$1 }'` ] # detect page not found answer |
336 | mv $(DIR_TMP)/$(notdir $@) $(DIR_DL) | |
337 | endef | |
338 | ||
339 | define MD5 | |
340 | # error mean file signature don't match the one in lfs script | |
341 | [ "$($@_MD5)" = `md5sum $(DIR_DL)/$@ | awk '{ print $$1 }'` ] | |
342 | echo "$@ checksum OK" | |
343 | endef | |
a8b159e7 MT |
344 | |
345 | define PAK | |
5c8cfc99 | 346 | # Bringing the files to their right place. |
258bc2e3 | 347 | @rm -rf $(DIR_TMP_PAK) && mkdir -p $(DIR_TMP_PAK) |
4c7fa778 | 348 | if [ -e "/usr/src/src/paks/$(PROG)" ]; then \ |
3489ebac | 349 | cp -f /usr/src/src/paks/$(PROG)/{,un}install.sh /usr/src/src/paks/$(PROG)/update.sh \ |
258bc2e3 | 350 | $(DIR_TMP_PAK); \ |
3489ebac MT |
351 | else \ |
352 | cp -f /usr/src/src/paks/default/{,un}install.sh /usr/src/src/paks/default/update.sh \ | |
258bc2e3 | 353 | $(DIR_TMP_PAK); \ |
3489ebac | 354 | fi |
dc7d6b20 | 355 | for i in $(DIR_SRC)/config/rootfiles/packages/{$(BUILD_ARCH),}/$(PROG); do \ |
5596077b | 356 | if [ -e "$${i}" ]; then \ |
258bc2e3 | 357 | cp -v $${i} $(DIR_TMP_PAK)/ROOTFILES; \ |
5596077b MT |
358 | break; \ |
359 | fi; \ | |
360 | done | |
258bc2e3 MT |
361 | |
362 | # Replace variables in rootfiles | |
363 | sed -i $(DIR_TMP_PAK)/ROOTFILES \ | |
364 | -e 's/BUILDTARGET/$(BUILDTARGET)/g' \ | |
365 | -e 's/KVER/$(KVER)/g' \ | |
289a86a3 | 366 | -e 's/xxxMACHINExxx/$(BUILD_ARCH)/g' |
258bc2e3 MT |
367 | |
368 | # Replace variables in scripts | |
369 | sed -i $(DIR_TMP_PAK)/install.sh \ | |
370 | -e 's/xxxKVERxxx/$(KVER)/g' | |
371 | ||
372 | # Make scripts executable | |
373 | chmod 754 $(DIR_TMP_PAK)/{{,un}install,update}.sh | |
374 | ||
375 | # Collect all files | |
376 | rm -rf $(DIR_TMP_PAK)/root && mkdir -p $(DIR_TMP_PAK)/root | |
377 | tar -c --exclude='#*' --exclude='proc/*' --exclude='dev/pts/*' --exclude='tmp/*' \ | |
378 | --exclude='__pycache__' \ | |
6a8be3e1 MT |
379 | -C / --files-from=$(DIR_TMP_PAK)/ROOTFILES | tar -x -C $(DIR_TMP_PAK)/root; \ |
380 | exit $${PIPESTATUS[0]} | |
258bc2e3 MT |
381 | |
382 | # Compress tarball | |
383 | cd $(DIR_TMP_PAK)/root && tar cf - * | xz $(XZ_OPT) > $(DIR_TMP_PAK)/files.tar.xz | |
384 | ||
385 | # Cleanup temporary files | |
386 | rm -rf $(DIR_TMP_PAK)/root | |
387 | ||
388 | # Remove any commented lines | |
389 | sed -i $(DIR_TMP_PAK)/ROOTFILES -e "/^#/d" | |
390 | ||
391 | # Make package | |
392 | cd $(DIR_TMP_PAK) && tar cf /install/packages/$(PROG)-$(VER)-$(PAK_VER).ipfire * | |
393 | ||
394 | # Cleanup | |
395 | rm -rf $(DIR_TMP_PAK) | |
396 | ||
397 | # Create meta file | |
398 | sed \ | |
399 | -e "s/NAME/$(PROG)/g" \ | |
400 | -e "s/VER/$(VER)/g" \ | |
401 | -e "s/RELEASE/$(PAK_VER)/g" \ | |
402 | -e "s/DEPS/$(DEPS)/g" \ | |
702b59cd | 403 | -e "s/SIZE/$$(stat --format=%s /install/packages/$(PROG)-$(VER)-$(PAK_VER).ipfire)/g" \ |
5c8cfc99 | 404 | < /usr/src/src/pakfire/meta > /install/packages/meta-$(PROG) |
a8b159e7 | 405 | endef |
38888b3d JS |
406 | |
407 | define INSTALL_INITSCRIPT | |
408 | install -m 754 -v $(DIR_SRC)/src/initscripts/packages/$(1) /etc/rc.d/init.d/$(1) | |
409 | endef | |
410 | ||
1ebdfb0e | 411 | ifeq "$(BUILD_ARCH)" "$(filter $(BUILD_ARCH),aarch64 riscv64)" |
0d1df90f MT |
412 | define UPDATE_AUTOMAKE |
413 | for i in $$(find $(DIR_APP) -name config.guess -o -name config.sub); do \ | |
5c36c8dd MT |
414 | cp -vf /usr/share/automake*/$$(basename $${i}) $${i} || \ |
415 | cp -vf $(TOOLS_DIR)/share/automake*/$$(basename $${i}) $${i}; \ | |
0d1df90f MT |
416 | done |
417 | endef | |
418 | endif | |
419 | ||
38888b3d JS |
420 | test: |
421 | $(call INSTALL_INITSCRIPT,hostapd) |