blob: 511bb0f878fa8280523f7832e2cd6c70a87e6cef [file] [log] [blame]
Sasha Smundakb051c4e2020-11-05 20:45:07 -08001// Copyright 2021 Google LLC
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://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,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
15package mk2rbc
16
17import (
18 "bytes"
Sasha Smundak6609ba72021-07-22 18:32:56 -070019 "io/fs"
20 "path/filepath"
Sasha Smundakb051c4e2020-11-05 20:45:07 -080021 "strings"
22 "testing"
23)
24
25var testCases = []struct {
26 desc string
27 mkname string
28 in string
29 expected string
30}{
31 {
32 desc: "Comment",
33 mkname: "product.mk",
34 in: `
35# Comment
36# FOO= a\
37 b
38`,
39 expected: `# Comment
40# FOO= a
41# b
42load("//build/make/core:product_config.rbc", "rblf")
43
44def init(g, handle):
45 cfg = rblf.cfg(handle)
46`,
47 },
48 {
49 desc: "Name conversion",
50 mkname: "path/bar-baz.mk",
51 in: `
52# Comment
53`,
54 expected: `# Comment
55load("//build/make/core:product_config.rbc", "rblf")
56
57def init(g, handle):
58 cfg = rblf.cfg(handle)
59`,
60 },
61 {
62 desc: "Item variable",
63 mkname: "pixel3.mk",
64 in: `
65PRODUCT_NAME := Pixel 3
66PRODUCT_MODEL :=
67local_var = foo
68`,
69 expected: `load("//build/make/core:product_config.rbc", "rblf")
70
71def init(g, handle):
72 cfg = rblf.cfg(handle)
73 cfg["PRODUCT_NAME"] = "Pixel 3"
74 cfg["PRODUCT_MODEL"] = ""
75 _local_var = "foo"
76`,
77 },
78 {
79 desc: "List variable",
80 mkname: "pixel4.mk",
81 in: `
82PRODUCT_PACKAGES = package1 package2
83PRODUCT_COPY_FILES += file2:target
84PRODUCT_PACKAGES += package3
85PRODUCT_COPY_FILES =
86`,
87 expected: `load("//build/make/core:product_config.rbc", "rblf")
88
89def init(g, handle):
90 cfg = rblf.cfg(handle)
91 cfg["PRODUCT_PACKAGES"] = [
92 "package1",
93 "package2",
94 ]
95 rblf.setdefault(handle, "PRODUCT_COPY_FILES")
96 cfg["PRODUCT_COPY_FILES"] += ["file2:target"]
97 cfg["PRODUCT_PACKAGES"] += ["package3"]
98 cfg["PRODUCT_COPY_FILES"] = []
99`,
100 },
101 {
102 desc: "Unknown function",
103 mkname: "product.mk",
104 in: `
Sasha Smundak6609ba72021-07-22 18:32:56 -0700105PRODUCT_NAME := $(call foo1, bar)
106PRODUCT_NAME := $(call foo0)
Sasha Smundakb051c4e2020-11-05 20:45:07 -0800107`,
Sasha Smundak6609ba72021-07-22 18:32:56 -0700108 expected: `# MK2RBC TRANSLATION ERROR: cannot handle invoking foo1
109# PRODUCT_NAME := $(call foo1, bar)
110# MK2RBC TRANSLATION ERROR: cannot handle invoking foo0
111# PRODUCT_NAME := $(call foo0)
Sasha Smundakb051c4e2020-11-05 20:45:07 -0800112load("//build/make/core:product_config.rbc", "rblf")
113
114def init(g, handle):
115 cfg = rblf.cfg(handle)
116 rblf.warning("product.mk", "partially successful conversion")
117`,
118 },
119 {
120 desc: "Inherit configuration always",
121 mkname: "product.mk",
122 in: `
Sasha Smundakb051c4e2020-11-05 20:45:07 -0800123$(call inherit-product, part.mk)
Sasha Smundak868c5e32021-09-23 16:20:58 -0700124ifdef PRODUCT_NAME
125$(call inherit-product, part1.mk)
Sasha Smundakb051c4e2020-11-05 20:45:07 -0800126else # Comment
Sasha Smundak868c5e32021-09-23 16:20:58 -0700127$(call inherit-product, $(LOCAL_PATH)/part1.mk)
Sasha Smundakb051c4e2020-11-05 20:45:07 -0800128endif
129`,
130 expected: `load("//build/make/core:product_config.rbc", "rblf")
131load(":part.star", _part_init = "init")
Sasha Smundak868c5e32021-09-23 16:20:58 -0700132load(":part1.star|init", _part1_init = "init")
Sasha Smundakb051c4e2020-11-05 20:45:07 -0800133
134def init(g, handle):
135 cfg = rblf.cfg(handle)
Sasha Smundak868c5e32021-09-23 16:20:58 -0700136 rblf.inherit(handle, "part", _part_init)
Sasha Smundakb051c4e2020-11-05 20:45:07 -0800137 if g.get("PRODUCT_NAME") != None:
Sasha Smundak868c5e32021-09-23 16:20:58 -0700138 rblf.inherit(handle, "part1", _part1_init)
Sasha Smundakb051c4e2020-11-05 20:45:07 -0800139 else:
140 # Comment
Sasha Smundak868c5e32021-09-23 16:20:58 -0700141 rblf.inherit(handle, "part1", _part1_init)
Sasha Smundakb051c4e2020-11-05 20:45:07 -0800142`,
143 },
144 {
145 desc: "Inherit configuration if it exists",
146 mkname: "product.mk",
147 in: `
148$(call inherit-product-if-exists, part.mk)
149`,
150 expected: `load("//build/make/core:product_config.rbc", "rblf")
151load(":part.star|init", _part_init = "init")
152
153def init(g, handle):
154 cfg = rblf.cfg(handle)
Sasha Smundak6609ba72021-07-22 18:32:56 -0700155 if _part_init:
Sasha Smundakb051c4e2020-11-05 20:45:07 -0800156 rblf.inherit(handle, "part", _part_init)
157`,
158 },
159
160 {
161 desc: "Include configuration",
162 mkname: "product.mk",
163 in: `
Sasha Smundakb051c4e2020-11-05 20:45:07 -0800164include part.mk
Sasha Smundak868c5e32021-09-23 16:20:58 -0700165ifdef PRODUCT_NAME
166include part1.mk
Sasha Smundakb051c4e2020-11-05 20:45:07 -0800167else
Sasha Smundak868c5e32021-09-23 16:20:58 -0700168-include $(LOCAL_PATH)/part1.mk)
Sasha Smundakb051c4e2020-11-05 20:45:07 -0800169endif
170`,
171 expected: `load("//build/make/core:product_config.rbc", "rblf")
Sasha Smundak868c5e32021-09-23 16:20:58 -0700172load(":part.star", _part_init = "init")
173load(":part1.star|init", _part1_init = "init")
Sasha Smundakb051c4e2020-11-05 20:45:07 -0800174
175def init(g, handle):
176 cfg = rblf.cfg(handle)
Sasha Smundak868c5e32021-09-23 16:20:58 -0700177 _part_init(g, handle)
Sasha Smundakb051c4e2020-11-05 20:45:07 -0800178 if g.get("PRODUCT_NAME") != None:
Sasha Smundak868c5e32021-09-23 16:20:58 -0700179 _part1_init(g, handle)
Sasha Smundakb051c4e2020-11-05 20:45:07 -0800180 else:
Sasha Smundak868c5e32021-09-23 16:20:58 -0700181 if _part1_init != None:
182 _part1_init(g, handle)
Sasha Smundakb051c4e2020-11-05 20:45:07 -0800183`,
184 },
185
186 {
187 desc: "Synonymous inherited configurations",
188 mkname: "path/product.mk",
189 in: `
Sasha Smundak6609ba72021-07-22 18:32:56 -0700190$(call inherit-product, */font.mk)
Sasha Smundakb051c4e2020-11-05 20:45:07 -0800191`,
192 expected: `load("//build/make/core:product_config.rbc", "rblf")
193load("//foo:font.star", _font_init = "init")
194load("//bar:font.star", _font1_init = "init")
195
196def init(g, handle):
197 cfg = rblf.cfg(handle)
198 rblf.inherit(handle, "foo/font", _font_init)
199 rblf.inherit(handle, "bar/font", _font1_init)
200`,
201 },
202 {
203 desc: "Directive define",
204 mkname: "product.mk",
205 in: `
206define some-macro
207 $(info foo)
208endef
209`,
210 expected: `# MK2RBC TRANSLATION ERROR: define is not supported: some-macro
211# define some-macro
212# $(info foo)
213# endef
214load("//build/make/core:product_config.rbc", "rblf")
215
216def init(g, handle):
217 cfg = rblf.cfg(handle)
218 rblf.warning("product.mk", "partially successful conversion")
219`,
220 },
221 {
222 desc: "Ifdef",
223 mkname: "product.mk",
224 in: `
225ifdef PRODUCT_NAME
226 PRODUCT_NAME = gizmo
227else
228endif
229`,
230 expected: `load("//build/make/core:product_config.rbc", "rblf")
231
232def init(g, handle):
233 cfg = rblf.cfg(handle)
234 if g.get("PRODUCT_NAME") != None:
235 cfg["PRODUCT_NAME"] = "gizmo"
236 else:
237 pass
238`,
239 },
240 {
241 desc: "Simple functions",
242 mkname: "product.mk",
243 in: `
244$(warning this is the warning)
245$(warning)
246$(info this is the info)
247$(error this is the error)
248PRODUCT_NAME:=$(shell echo *)
249`,
250 expected: `load("//build/make/core:product_config.rbc", "rblf")
251
252def init(g, handle):
253 cfg = rblf.cfg(handle)
254 rblf.mkwarning("product.mk", "this is the warning")
255 rblf.mkwarning("product.mk", "")
256 rblf.mkinfo("product.mk", "this is the info")
257 rblf.mkerror("product.mk", "this is the error")
258 cfg["PRODUCT_NAME"] = rblf.shell("echo *")
259`,
260 },
261 {
262 desc: "Empty if",
263 mkname: "product.mk",
264 in: `
265ifdef PRODUCT_NAME
266# Comment
Sasha Smundak6609ba72021-07-22 18:32:56 -0700267else
Sasha Smundak02183cf2021-08-16 13:36:11 -0700268 TARGET_COPY_OUT_RECOVERY := foo
Sasha Smundakb051c4e2020-11-05 20:45:07 -0800269endif
270`,
271 expected: `load("//build/make/core:product_config.rbc", "rblf")
272
273def init(g, handle):
274 cfg = rblf.cfg(handle)
275 if g.get("PRODUCT_NAME") != None:
276 # Comment
277 pass
Sasha Smundak6609ba72021-07-22 18:32:56 -0700278 else:
Sasha Smundak02183cf2021-08-16 13:36:11 -0700279 # MK2RBC TRANSLATION ERROR: cannot set predefined variable TARGET_COPY_OUT_RECOVERY to "foo", its value should be "recovery"
Sasha Smundak6609ba72021-07-22 18:32:56 -0700280 pass
281 rblf.warning("product.mk", "partially successful conversion")
Sasha Smundakb051c4e2020-11-05 20:45:07 -0800282`,
283 },
284 {
285 desc: "if/else/endif",
286 mkname: "product.mk",
287 in: `
288ifndef PRODUCT_NAME
289 PRODUCT_NAME=gizmo1
290else
291 PRODUCT_NAME=gizmo2
292endif
293`,
294 expected: `load("//build/make/core:product_config.rbc", "rblf")
295
296def init(g, handle):
297 cfg = rblf.cfg(handle)
298 if not g.get("PRODUCT_NAME") != None:
299 cfg["PRODUCT_NAME"] = "gizmo1"
300 else:
301 cfg["PRODUCT_NAME"] = "gizmo2"
302`,
303 },
304 {
305 desc: "else if",
306 mkname: "product.mk",
307 in: `
308ifdef PRODUCT_NAME
309 PRODUCT_NAME = gizmo
310else ifndef PRODUCT_PACKAGES # Comment
311endif
312 `,
313 expected: `load("//build/make/core:product_config.rbc", "rblf")
314
315def init(g, handle):
316 cfg = rblf.cfg(handle)
317 if g.get("PRODUCT_NAME") != None:
318 cfg["PRODUCT_NAME"] = "gizmo"
319 elif not g.get("PRODUCT_PACKAGES") != None:
320 # Comment
321 pass
322`,
323 },
324 {
325 desc: "ifeq / ifneq",
326 mkname: "product.mk",
327 in: `
328ifeq (aosp_arm, $(TARGET_PRODUCT))
329 PRODUCT_MODEL = pix2
330else
331 PRODUCT_MODEL = pix21
332endif
333ifneq (aosp_x86, $(TARGET_PRODUCT))
334 PRODUCT_MODEL = pix3
335endif
336`,
337 expected: `load("//build/make/core:product_config.rbc", "rblf")
338
339def init(g, handle):
340 cfg = rblf.cfg(handle)
341 if "aosp_arm" == g["TARGET_PRODUCT"]:
342 cfg["PRODUCT_MODEL"] = "pix2"
343 else:
344 cfg["PRODUCT_MODEL"] = "pix21"
345 if "aosp_x86" != g["TARGET_PRODUCT"]:
346 cfg["PRODUCT_MODEL"] = "pix3"
347`,
348 },
349 {
350 desc: "Check filter result",
351 mkname: "product.mk",
352 in: `
353ifeq (,$(filter userdebug eng, $(TARGET_BUILD_VARIANT)))
354endif
355ifneq (,$(filter userdebug,$(TARGET_BUILD_VARIANT))
356endif
357ifneq (,$(filter plaf,$(PLATFORM_LIST)))
358endif
359ifeq ($(TARGET_BUILD_VARIANT), $(filter $(TARGET_BUILD_VARIANT), userdebug eng))
360endif
Sasha Smundak0554d762021-07-08 18:26:12 -0700361ifneq (,$(filter true, $(v1)$(v2)))
362endif
Sasha Smundak5f463be2021-09-15 18:43:36 -0700363ifeq (,$(filter barbet coral%,$(TARGET_PRODUCT)))
364else ifneq (,$(filter barbet%,$(TARGET_PRODUCT)))
365endif
Sasha Smundakb051c4e2020-11-05 20:45:07 -0800366`,
367 expected: `load("//build/make/core:product_config.rbc", "rblf")
368
369def init(g, handle):
370 cfg = rblf.cfg(handle)
Sasha Smundak5f463be2021-09-15 18:43:36 -0700371 if not rblf.filter("userdebug eng", g["TARGET_BUILD_VARIANT"]):
Sasha Smundakb051c4e2020-11-05 20:45:07 -0800372 pass
Sasha Smundak5f463be2021-09-15 18:43:36 -0700373 if rblf.filter("userdebug", g["TARGET_BUILD_VARIANT"]):
Sasha Smundakb051c4e2020-11-05 20:45:07 -0800374 pass
375 if "plaf" in g.get("PLATFORM_LIST", []):
376 pass
377 if g["TARGET_BUILD_VARIANT"] in ["userdebug", "eng"]:
378 pass
Sasha Smundak5f463be2021-09-15 18:43:36 -0700379 if rblf.filter("true", "%s%s" % (_v1, _v2)):
380 pass
381 if not rblf.filter("barbet coral%", g["TARGET_PRODUCT"]):
382 pass
383 elif rblf.filter("barbet%", g["TARGET_PRODUCT"]):
Sasha Smundak0554d762021-07-08 18:26:12 -0700384 pass
Sasha Smundakb051c4e2020-11-05 20:45:07 -0800385`,
386 },
387 {
388 desc: "Get filter result",
389 mkname: "product.mk",
390 in: `
391PRODUCT_LIST2=$(filter-out %/foo.ko,$(wildcard path/*.ko))
392`,
393 expected: `load("//build/make/core:product_config.rbc", "rblf")
394
395def init(g, handle):
396 cfg = rblf.cfg(handle)
397 cfg["PRODUCT_LIST2"] = rblf.filter_out("%/foo.ko", rblf.expand_wildcard("path/*.ko"))
398`,
399 },
400 {
401 desc: "filter $(VAR), values",
402 mkname: "product.mk",
403 in: `
404ifeq (,$(filter $(TARGET_PRODUCT), yukawa_gms yukawa_sei510_gms)
405 ifneq (,$(filter $(TARGET_PRODUCT), yukawa_gms)
406 endif
407endif
408
409`,
410 expected: `load("//build/make/core:product_config.rbc", "rblf")
411
412def init(g, handle):
413 cfg = rblf.cfg(handle)
414 if g["TARGET_PRODUCT"] not in ["yukawa_gms", "yukawa_sei510_gms"]:
Sasha Smundak0554d762021-07-08 18:26:12 -0700415 if g["TARGET_PRODUCT"] == "yukawa_gms":
Sasha Smundakb051c4e2020-11-05 20:45:07 -0800416 pass
417`,
418 },
419 {
Sasha Smundak0554d762021-07-08 18:26:12 -0700420 desc: "filter $(V1), $(V2)",
421 mkname: "product.mk",
422 in: `
423ifneq (, $(filter $(PRODUCT_LIST), $(TARGET_PRODUCT)))
424endif
425`,
426 expected: `load("//build/make/core:product_config.rbc", "rblf")
427
428def init(g, handle):
429 cfg = rblf.cfg(handle)
Sasha Smundak468e11f2021-08-26 09:10:23 -0700430 if rblf.filter(g.get("PRODUCT_LIST", []), g["TARGET_PRODUCT"]):
Sasha Smundak0554d762021-07-08 18:26:12 -0700431 pass
432`,
433 },
434 {
Sasha Smundakb051c4e2020-11-05 20:45:07 -0800435 desc: "ifeq",
436 mkname: "product.mk",
437 in: `
438ifeq (aosp, $(TARGET_PRODUCT)) # Comment
439else ifneq (, $(TARGET_PRODUCT))
440endif
441`,
442 expected: `load("//build/make/core:product_config.rbc", "rblf")
443
444def init(g, handle):
445 cfg = rblf.cfg(handle)
446 if "aosp" == g["TARGET_PRODUCT"]:
447 # Comment
448 pass
449 elif g["TARGET_PRODUCT"]:
450 pass
451`,
452 },
453 {
454 desc: "Nested if",
455 mkname: "product.mk",
456 in: `
457ifdef PRODUCT_NAME
458 PRODUCT_PACKAGES = pack-if0
459 ifdef PRODUCT_MODEL
460 PRODUCT_PACKAGES = pack-if-if
461 else ifdef PRODUCT_NAME
462 PRODUCT_PACKAGES = pack-if-elif
463 else
464 PRODUCT_PACKAGES = pack-if-else
465 endif
466 PRODUCT_PACKAGES = pack-if
467else ifneq (,$(TARGET_PRODUCT))
468 PRODUCT_PACKAGES = pack-elif
469else
470 PRODUCT_PACKAGES = pack-else
471endif
472`,
473 expected: `load("//build/make/core:product_config.rbc", "rblf")
474
475def init(g, handle):
476 cfg = rblf.cfg(handle)
477 if g.get("PRODUCT_NAME") != None:
478 cfg["PRODUCT_PACKAGES"] = ["pack-if0"]
479 if g.get("PRODUCT_MODEL") != None:
480 cfg["PRODUCT_PACKAGES"] = ["pack-if-if"]
481 elif g.get("PRODUCT_NAME") != None:
482 cfg["PRODUCT_PACKAGES"] = ["pack-if-elif"]
483 else:
484 cfg["PRODUCT_PACKAGES"] = ["pack-if-else"]
485 cfg["PRODUCT_PACKAGES"] = ["pack-if"]
486 elif g["TARGET_PRODUCT"]:
487 cfg["PRODUCT_PACKAGES"] = ["pack-elif"]
488 else:
489 cfg["PRODUCT_PACKAGES"] = ["pack-else"]
490`,
491 },
492 {
493 desc: "Wildcard",
494 mkname: "product.mk",
495 in: `
496ifeq (,$(wildcard foo.mk))
497endif
498ifneq (,$(wildcard foo*.mk))
499endif
500`,
501 expected: `load("//build/make/core:product_config.rbc", "rblf")
502
503def init(g, handle):
504 cfg = rblf.cfg(handle)
505 if not rblf.file_exists("foo.mk"):
506 pass
507 if rblf.file_wildcard_exists("foo*.mk"):
508 pass
509`,
510 },
511 {
512 desc: "ifneq $(X),true",
513 mkname: "product.mk",
514 in: `
515ifneq ($(VARIABLE),true)
516endif
517`,
518 expected: `load("//build/make/core:product_config.rbc", "rblf")
519
520def init(g, handle):
521 cfg = rblf.cfg(handle)
522 if g.get("VARIABLE", "") != "true":
523 pass
524`,
525 },
526 {
527 desc: "Const neq",
528 mkname: "product.mk",
529 in: `
530ifneq (1,0)
531endif
532`,
533 expected: `load("//build/make/core:product_config.rbc", "rblf")
534
535def init(g, handle):
536 cfg = rblf.cfg(handle)
537 if "1" != "0":
538 pass
539`,
540 },
541 {
542 desc: "is-board calls",
543 mkname: "product.mk",
544 in: `
545ifeq ($(call is-board-platform-in-list,msm8998), true)
546else ifneq ($(call is-board-platform,copper),true)
547else ifneq ($(call is-vendor-board-platform,QCOM),true)
548else ifeq ($(call is-product-in-list, $(PLATFORM_LIST)), true)
549endif
550`,
551 expected: `load("//build/make/core:product_config.rbc", "rblf")
552
553def init(g, handle):
554 cfg = rblf.cfg(handle)
555 if g.get("TARGET_BOARD_PLATFORM", "") in ["msm8998"]:
556 pass
557 elif g.get("TARGET_BOARD_PLATFORM", "") != "copper":
558 pass
559 elif g.get("TARGET_BOARD_PLATFORM", "") not in g["QCOM_BOARD_PLATFORMS"]:
560 pass
561 elif g["TARGET_PRODUCT"] in g.get("PLATFORM_LIST", []):
562 pass
563`,
564 },
565 {
Sasha Smundak3a9b8e82021-08-25 14:11:04 -0700566 desc: "new is-board calls",
567 mkname: "product.mk",
568 in: `
569ifneq (,$(call is-board-platform-in-list2,msm8998 $(X))
570else ifeq (,$(call is-board-platform2,copper)
571else ifneq (,$(call is-vendor-board-qcom))
572endif
573`,
574 expected: `load("//build/make/core:product_config.rbc", "rblf")
575
576def init(g, handle):
577 cfg = rblf.cfg(handle)
578 if rblf.board_platform_in(g, "msm8998 %s" % g.get("X", "")):
579 pass
580 elif not rblf.board_platform_is(g, "copper"):
581 pass
582 elif g.get("TARGET_BOARD_PLATFORM", "") not in g["QCOM_BOARD_PLATFORMS"]:
583 pass
584`,
585 },
586 {
Sasha Smundakb051c4e2020-11-05 20:45:07 -0800587 desc: "findstring call",
588 mkname: "product.mk",
589 in: `
590ifneq ($(findstring foo,$(PRODUCT_PACKAGES)),)
591endif
592`,
593 expected: `load("//build/make/core:product_config.rbc", "rblf")
594
595def init(g, handle):
596 cfg = rblf.cfg(handle)
597 if (cfg.get("PRODUCT_PACKAGES", [])).find("foo") != -1:
598 pass
599`,
600 },
601 {
602 desc: "rhs call",
603 mkname: "product.mk",
604 in: `
605PRODUCT_COPY_FILES = $(call add-to-product-copy-files-if-exists, path:distpath) \
606 $(call find-copy-subdir-files, *, fromdir, todir) $(wildcard foo.*)
607`,
608 expected: `load("//build/make/core:product_config.rbc", "rblf")
609
610def init(g, handle):
611 cfg = rblf.cfg(handle)
612 cfg["PRODUCT_COPY_FILES"] = (rblf.copy_if_exists("path:distpath") +
613 rblf.find_and_copy("*", "fromdir", "todir") +
614 rblf.expand_wildcard("foo.*"))
615`,
616 },
617 {
618 desc: "inferred type",
619 mkname: "product.mk",
620 in: `
621HIKEY_MODS := $(wildcard foo/*.ko)
622BOARD_VENDOR_KERNEL_MODULES += $(HIKEY_MODS)
623`,
624 expected: `load("//build/make/core:product_config.rbc", "rblf")
625
626def init(g, handle):
627 cfg = rblf.cfg(handle)
628 g["HIKEY_MODS"] = rblf.expand_wildcard("foo/*.ko")
629 g.setdefault("BOARD_VENDOR_KERNEL_MODULES", [])
630 g["BOARD_VENDOR_KERNEL_MODULES"] += g["HIKEY_MODS"]
631`,
632 },
633 {
634 desc: "list with vars",
635 mkname: "product.mk",
636 in: `
637PRODUCT_COPY_FILES += path1:$(TARGET_PRODUCT)/path1 $(PRODUCT_MODEL)/path2:$(TARGET_PRODUCT)/path2
638`,
639 expected: `load("//build/make/core:product_config.rbc", "rblf")
640
641def init(g, handle):
642 cfg = rblf.cfg(handle)
643 rblf.setdefault(handle, "PRODUCT_COPY_FILES")
644 cfg["PRODUCT_COPY_FILES"] += (("path1:%s/path1" % g["TARGET_PRODUCT"]).split() +
645 ("%s/path2:%s/path2" % (cfg.get("PRODUCT_MODEL", ""), g["TARGET_PRODUCT"])).split())
646`,
647 },
648 {
649 desc: "misc calls",
650 mkname: "product.mk",
651 in: `
652$(call enforce-product-packages-exist,)
653$(call enforce-product-packages-exist, foo)
654$(call require-artifacts-in-path, foo, bar)
655$(call require-artifacts-in-path-relaxed, foo, bar)
656`,
657 expected: `load("//build/make/core:product_config.rbc", "rblf")
658
659def init(g, handle):
660 cfg = rblf.cfg(handle)
661 rblf.enforce_product_packages_exist("")
662 rblf.enforce_product_packages_exist("foo")
663 rblf.require_artifacts_in_path("foo", "bar")
664 rblf.require_artifacts_in_path_relaxed("foo", "bar")
665`,
666 },
667 {
668 desc: "list with functions",
669 mkname: "product.mk",
670 in: `
671PRODUCT_COPY_FILES := $(call find-copy-subdir-files,*.kl,from1,to1) \
672 $(call find-copy-subdir-files,*.kc,from2,to2) \
673 foo bar
674`,
675 expected: `load("//build/make/core:product_config.rbc", "rblf")
676
677def init(g, handle):
678 cfg = rblf.cfg(handle)
679 cfg["PRODUCT_COPY_FILES"] = (rblf.find_and_copy("*.kl", "from1", "to1") +
680 rblf.find_and_copy("*.kc", "from2", "to2") +
681 [
682 "foo",
683 "bar",
684 ])
685`,
686 },
687 {
688 desc: "Text functions",
689 mkname: "product.mk",
690 in: `
691PRODUCT_COPY_FILES := $(addprefix pfx-,a b c)
692PRODUCT_COPY_FILES := $(addsuffix .sff, a b c)
693PRODUCT_NAME := $(word 1, $(subst ., ,$(TARGET_BOARD_PLATFORM)))
Sasha Smundak94b41c72021-07-12 18:30:42 -0700694$(info $(patsubst %.pub,%,$(PRODUCT_ADB_KEYS)))
Sasha Smundak16e07732021-07-23 11:38:23 -0700695$(info $(dir foo/bar))
696$(info $(firstword $(PRODUCT_COPY_FILES)))
697$(info $(lastword $(PRODUCT_COPY_FILES)))
698$(info $(dir $(lastword $(MAKEFILE_LIST))))
699$(info $(dir $(lastword $(PRODUCT_COPY_FILES))))
700$(info $(dir $(lastword $(foobar))))
701$(info $(abspath foo/bar))
702$(info $(notdir foo/bar))
Sasha Smundak3deb9682021-07-26 18:42:25 -0700703$(call add_soong_config_namespace,snsconfig)
704$(call add_soong_config_var_value,snsconfig,imagetype,odm_image)
Sasha Smundak65b547e2021-09-17 15:35:41 -0700705$(call soong_config_set, snsconfig, foo, foo_value)
706$(call soong_config_append, snsconfig, bar, bar_value)
Sasha Smundak3deb9682021-07-26 18:42:25 -0700707PRODUCT_COPY_FILES := $(call copy-files,$(wildcard foo*.mk),etc)
Sasha Smundak04453082021-08-17 18:14:41 -0700708PRODUCT_COPY_FILES := $(call product-copy-files-by-pattern,from/%,to/%,a b c)
Sasha Smundakb051c4e2020-11-05 20:45:07 -0800709`,
710 expected: `load("//build/make/core:product_config.rbc", "rblf")
711
712def init(g, handle):
713 cfg = rblf.cfg(handle)
714 cfg["PRODUCT_COPY_FILES"] = rblf.addprefix("pfx-", "a b c")
715 cfg["PRODUCT_COPY_FILES"] = rblf.addsuffix(".sff", "a b c")
716 cfg["PRODUCT_NAME"] = ((g.get("TARGET_BOARD_PLATFORM", "")).replace(".", " ")).split()[0]
Sasha Smundak94b41c72021-07-12 18:30:42 -0700717 rblf.mkinfo("product.mk", rblf.mkpatsubst("%.pub", "%", g.get("PRODUCT_ADB_KEYS", "")))
Sasha Smundak16e07732021-07-23 11:38:23 -0700718 rblf.mkinfo("product.mk", rblf.dir("foo/bar"))
719 rblf.mkinfo("product.mk", cfg["PRODUCT_COPY_FILES"][0])
720 rblf.mkinfo("product.mk", cfg["PRODUCT_COPY_FILES"][-1])
721 rblf.mkinfo("product.mk", rblf.dir("product.mk"))
722 rblf.mkinfo("product.mk", rblf.dir(cfg["PRODUCT_COPY_FILES"][-1]))
723 rblf.mkinfo("product.mk", rblf.dir((_foobar).split()[-1]))
724 rblf.mkinfo("product.mk", rblf.abspath("foo/bar"))
725 rblf.mkinfo("product.mk", rblf.notdir("foo/bar"))
Sasha Smundak65b547e2021-09-17 15:35:41 -0700726 rblf.soong_config_namespace(g, "snsconfig")
727 rblf.soong_config_set(g, "snsconfig", "imagetype", "odm_image")
728 rblf.soong_config_set(g, "snsconfig", "foo", "foo_value")
729 rblf.soong_config_append(g, "snsconfig", "bar", "bar_value")
Sasha Smundak3deb9682021-07-26 18:42:25 -0700730 cfg["PRODUCT_COPY_FILES"] = rblf.copy_files(rblf.expand_wildcard("foo*.mk"), "etc")
Sasha Smundak04453082021-08-17 18:14:41 -0700731 cfg["PRODUCT_COPY_FILES"] = rblf.product_copy_files_by_pattern("from/%", "to/%", "a b c")
Sasha Smundakb051c4e2020-11-05 20:45:07 -0800732`,
733 },
734 {
Sasha Smundak9d011ab2021-07-09 16:00:57 -0700735 desc: "subst in list",
736 mkname: "product.mk",
737 in: `
738files = $(call find-copy-subdir-files,*,from,to)
739PRODUCT_COPY_FILES += $(subst foo,bar,$(files))
740`,
741 expected: `load("//build/make/core:product_config.rbc", "rblf")
742
743def init(g, handle):
744 cfg = rblf.cfg(handle)
745 _files = rblf.find_and_copy("*", "from", "to")
746 rblf.setdefault(handle, "PRODUCT_COPY_FILES")
747 cfg["PRODUCT_COPY_FILES"] += rblf.mksubst("foo", "bar", _files)
748`,
749 },
750 {
Sasha Smundakb051c4e2020-11-05 20:45:07 -0800751 desc: "assignment flavors",
752 mkname: "product.mk",
753 in: `
754PRODUCT_LIST1 := a
755PRODUCT_LIST2 += a
756PRODUCT_LIST1 += b
757PRODUCT_LIST2 += b
758PRODUCT_LIST3 ?= a
759PRODUCT_LIST1 = c
760PLATFORM_LIST += x
761PRODUCT_PACKAGES := $(PLATFORM_LIST)
762`,
763 expected: `load("//build/make/core:product_config.rbc", "rblf")
764
765def init(g, handle):
766 cfg = rblf.cfg(handle)
767 cfg["PRODUCT_LIST1"] = ["a"]
768 rblf.setdefault(handle, "PRODUCT_LIST2")
769 cfg["PRODUCT_LIST2"] += ["a"]
770 cfg["PRODUCT_LIST1"] += ["b"]
771 cfg["PRODUCT_LIST2"] += ["b"]
772 if cfg.get("PRODUCT_LIST3") == None:
773 cfg["PRODUCT_LIST3"] = ["a"]
774 cfg["PRODUCT_LIST1"] = ["c"]
775 g.setdefault("PLATFORM_LIST", [])
776 g["PLATFORM_LIST"] += ["x"]
777 cfg["PRODUCT_PACKAGES"] = g["PLATFORM_LIST"][:]
778`,
779 },
780 {
781 desc: "assigment flavors2",
782 mkname: "product.mk",
783 in: `
784PRODUCT_LIST1 = a
785ifeq (0,1)
786 PRODUCT_LIST1 += b
787 PRODUCT_LIST2 += b
788endif
789PRODUCT_LIST1 += c
790PRODUCT_LIST2 += c
791`,
792 expected: `load("//build/make/core:product_config.rbc", "rblf")
793
794def init(g, handle):
795 cfg = rblf.cfg(handle)
796 cfg["PRODUCT_LIST1"] = ["a"]
797 if "0" == "1":
798 cfg["PRODUCT_LIST1"] += ["b"]
799 rblf.setdefault(handle, "PRODUCT_LIST2")
800 cfg["PRODUCT_LIST2"] += ["b"]
801 cfg["PRODUCT_LIST1"] += ["c"]
802 rblf.setdefault(handle, "PRODUCT_LIST2")
803 cfg["PRODUCT_LIST2"] += ["c"]
804`,
805 },
806 {
Sasha Smundak3deb9682021-07-26 18:42:25 -0700807 desc: "soong namespace assignments",
808 mkname: "product.mk",
809 in: `
810SOONG_CONFIG_NAMESPACES += cvd
811SOONG_CONFIG_cvd += launch_configs
Sasha Smundak65b547e2021-09-17 15:35:41 -0700812SOONG_CONFIG_cvd_launch_configs = cvd_config_auto.json
Sasha Smundak3deb9682021-07-26 18:42:25 -0700813SOONG_CONFIG_cvd += grub_config
814SOONG_CONFIG_cvd_grub_config += grub.cfg
Sasha Smundak65b547e2021-09-17 15:35:41 -0700815x := $(SOONG_CONFIG_cvd_grub_config)
Sasha Smundak3deb9682021-07-26 18:42:25 -0700816`,
817 expected: `load("//build/make/core:product_config.rbc", "rblf")
818
819def init(g, handle):
820 cfg = rblf.cfg(handle)
Sasha Smundak65b547e2021-09-17 15:35:41 -0700821 rblf.soong_config_namespace(g, "cvd")
822 rblf.soong_config_set(g, "cvd", "launch_configs", "cvd_config_auto.json")
823 rblf.soong_config_append(g, "cvd", "grub_config", "grub.cfg")
824 # MK2RBC TRANSLATION ERROR: SOONG_CONFIG_ variables cannot be referenced: SOONG_CONFIG_cvd_grub_config
825 # x := $(SOONG_CONFIG_cvd_grub_config)
826 rblf.warning("product.mk", "partially successful conversion")
Sasha Smundak3deb9682021-07-26 18:42:25 -0700827`,
828 },
829 {
Sasha Smundakb051c4e2020-11-05 20:45:07 -0800830 desc: "string split",
831 mkname: "product.mk",
832 in: `
833PRODUCT_LIST1 = a
834local = b
835local += c
836FOO = d
837FOO += e
838PRODUCT_LIST1 += $(local)
839PRODUCT_LIST1 += $(FOO)
840`,
841 expected: `load("//build/make/core:product_config.rbc", "rblf")
842
843def init(g, handle):
844 cfg = rblf.cfg(handle)
845 cfg["PRODUCT_LIST1"] = ["a"]
846 _local = "b"
847 _local += " " + "c"
848 g["FOO"] = "d"
849 g["FOO"] += " " + "e"
850 cfg["PRODUCT_LIST1"] += (_local).split()
851 cfg["PRODUCT_LIST1"] += (g["FOO"]).split()
852`,
853 },
854 {
855 desc: "apex_jars",
856 mkname: "product.mk",
857 in: `
858PRODUCT_BOOT_JARS := $(ART_APEX_JARS) framework-minus-apex
859`,
860 expected: `load("//build/make/core:product_config.rbc", "rblf")
861
862def init(g, handle):
863 cfg = rblf.cfg(handle)
864 cfg["PRODUCT_BOOT_JARS"] = (g.get("ART_APEX_JARS", []) +
865 ["framework-minus-apex"])
866`,
867 },
868 {
869 desc: "strip function",
870 mkname: "product.mk",
871 in: `
872ifeq ($(filter hwaddress,$(PRODUCT_PACKAGES)),)
873 PRODUCT_PACKAGES := $(strip $(PRODUCT_PACKAGES) hwaddress)
874endif
875`,
876 expected: `load("//build/make/core:product_config.rbc", "rblf")
877
878def init(g, handle):
879 cfg = rblf.cfg(handle)
880 if "hwaddress" not in cfg.get("PRODUCT_PACKAGES", []):
881 cfg["PRODUCT_PACKAGES"] = (rblf.mkstrip("%s hwaddress" % " ".join(cfg.get("PRODUCT_PACKAGES", [])))).split()
882`,
883 },
884 {
885 desc: "strip func in condition",
886 mkname: "product.mk",
887 in: `
888ifneq ($(strip $(TARGET_VENDOR)),)
889endif
890`,
891 expected: `load("//build/make/core:product_config.rbc", "rblf")
892
893def init(g, handle):
894 cfg = rblf.cfg(handle)
Sasha Smundak0554d762021-07-08 18:26:12 -0700895 if rblf.mkstrip(g.get("TARGET_VENDOR", "")):
Sasha Smundakb051c4e2020-11-05 20:45:07 -0800896 pass
897`,
898 },
899 {
900 desc: "ref after set",
901 mkname: "product.mk",
902 in: `
903PRODUCT_ADB_KEYS:=value
904FOO := $(PRODUCT_ADB_KEYS)
905ifneq (,$(PRODUCT_ADB_KEYS))
906endif
907`,
908 expected: `load("//build/make/core:product_config.rbc", "rblf")
909
910def init(g, handle):
911 cfg = rblf.cfg(handle)
912 g["PRODUCT_ADB_KEYS"] = "value"
913 g["FOO"] = g["PRODUCT_ADB_KEYS"]
914 if g["PRODUCT_ADB_KEYS"]:
915 pass
916`,
917 },
918 {
919 desc: "ref before set",
920 mkname: "product.mk",
921 in: `
922V1 := $(PRODUCT_ADB_KEYS)
923ifeq (,$(PRODUCT_ADB_KEYS))
924 V2 := $(PRODUCT_ADB_KEYS)
925 PRODUCT_ADB_KEYS:=foo
926 V3 := $(PRODUCT_ADB_KEYS)
927endif`,
928 expected: `load("//build/make/core:product_config.rbc", "rblf")
929
930def init(g, handle):
931 cfg = rblf.cfg(handle)
932 g["V1"] = g.get("PRODUCT_ADB_KEYS", "")
933 if not g.get("PRODUCT_ADB_KEYS", ""):
934 g["V2"] = g.get("PRODUCT_ADB_KEYS", "")
935 g["PRODUCT_ADB_KEYS"] = "foo"
936 g["V3"] = g["PRODUCT_ADB_KEYS"]
937`,
938 },
Sasha Smundak6609ba72021-07-22 18:32:56 -0700939 {
940 desc: "Dynamic inherit path",
941 mkname: "product.mk",
942 in: `
Sasha Smundak6d852dd2021-09-27 20:34:39 -0700943MY_PATH:=foo
Sasha Smundak6609ba72021-07-22 18:32:56 -0700944$(call inherit-product,vendor/$(MY_PATH)/cfg.mk)
945`,
946 expected: `load("//build/make/core:product_config.rbc", "rblf")
947load("//vendor/foo1:cfg.star|init", _cfg_init = "init")
948load("//vendor/bar/baz:cfg.star|init", _cfg1_init = "init")
949
950def init(g, handle):
951 cfg = rblf.cfg(handle)
952 g["MY_PATH"] = "foo"
953 _entry = {
954 "vendor/foo1/cfg.mk": ("_cfg", _cfg_init),
955 "vendor/bar/baz/cfg.mk": ("_cfg1", _cfg1_init),
956 }.get("vendor/%s/cfg.mk" % g["MY_PATH"])
957 (_varmod, _varmod_init) = _entry if _entry else (None, None)
958 if not _varmod_init:
959 rblf.mkerror("cannot")
960 rblf.inherit(handle, _varmod, _varmod_init)
961`,
962 },
Sasha Smundak6d852dd2021-09-27 20:34:39 -0700963 {
964 desc: "Dynamic inherit with hint",
965 mkname: "product.mk",
966 in: `
967MY_PATH:=foo
968#RBC# include_top vendor/foo1
969$(call inherit-product,$(MY_PATH)/cfg.mk)
970`,
971 expected: `load("//build/make/core:product_config.rbc", "rblf")
972load("//vendor/foo1:cfg.star|init", _cfg_init = "init")
973
974def init(g, handle):
975 cfg = rblf.cfg(handle)
976 g["MY_PATH"] = "foo"
977 #RBC# include_top vendor/foo1
978 _entry = {
979 "vendor/foo1/cfg.mk": ("_cfg", _cfg_init),
980 }.get("%s/cfg.mk" % g["MY_PATH"])
981 (_varmod, _varmod_init) = _entry if _entry else (None, None)
982 if not _varmod_init:
983 rblf.mkerror("cannot")
984 rblf.inherit(handle, _varmod, _varmod_init)
985`,
986 },
Sasha Smundak2afb9d72021-10-24 15:16:59 -0700987 {
988 desc: "Ignore make rules",
989 mkname: "product.mk",
990 in: `
991foo: foo.c
992 gcc -o $@ $*`,
993 expected: `# MK2RBC TRANSLATION ERROR: unsupported line rule: foo: foo.c
994#gcc -o $@ $*
995# rule: foo: foo.c
996# gcc -o $@ $*
997load("//build/make/core:product_config.rbc", "rblf")
998
999def init(g, handle):
1000 cfg = rblf.cfg(handle)
1001 rblf.warning("product.mk", "partially successful conversion")
1002`,
1003 },
Sasha Smundakb051c4e2020-11-05 20:45:07 -08001004}
1005
1006var known_variables = []struct {
1007 name string
1008 class varClass
1009 starlarkType
1010}{
1011 {"PRODUCT_NAME", VarClassConfig, starlarkTypeString},
1012 {"PRODUCT_MODEL", VarClassConfig, starlarkTypeString},
1013 {"PRODUCT_PACKAGES", VarClassConfig, starlarkTypeList},
1014 {"PRODUCT_BOOT_JARS", VarClassConfig, starlarkTypeList},
1015 {"PRODUCT_COPY_FILES", VarClassConfig, starlarkTypeList},
1016 {"PRODUCT_IS_64BIT", VarClassConfig, starlarkTypeString},
1017 {"PRODUCT_LIST1", VarClassConfig, starlarkTypeList},
1018 {"PRODUCT_LIST2", VarClassConfig, starlarkTypeList},
1019 {"PRODUCT_LIST3", VarClassConfig, starlarkTypeList},
1020 {"TARGET_PRODUCT", VarClassSoong, starlarkTypeString},
1021 {"TARGET_BUILD_VARIANT", VarClassSoong, starlarkTypeString},
1022 {"TARGET_BOARD_PLATFORM", VarClassSoong, starlarkTypeString},
1023 {"QCOM_BOARD_PLATFORMS", VarClassSoong, starlarkTypeString},
1024 {"PLATFORM_LIST", VarClassSoong, starlarkTypeList}, // TODO(asmundak): make it local instead of soong
1025}
1026
Sasha Smundak6609ba72021-07-22 18:32:56 -07001027type testMakefileFinder struct {
1028 fs fs.FS
1029 root string
1030 files []string
1031}
1032
1033func (t *testMakefileFinder) Find(root string) []string {
1034 if t.files != nil || root == t.root {
1035 return t.files
1036 }
1037 t.files = make([]string, 0)
1038 fs.WalkDir(t.fs, root, func(path string, d fs.DirEntry, err error) error {
1039 if err != nil {
1040 return err
1041 }
1042 if d.IsDir() {
1043 base := filepath.Base(path)
1044 if base[0] == '.' && len(base) > 1 {
1045 return fs.SkipDir
1046 }
1047 return nil
1048 }
1049 if strings.HasSuffix(path, ".mk") {
1050 t.files = append(t.files, path)
1051 }
1052 return nil
1053 })
1054 return t.files
1055}
1056
Sasha Smundakb051c4e2020-11-05 20:45:07 -08001057func TestGood(t *testing.T) {
1058 for _, v := range known_variables {
1059 KnownVariables.NewVariable(v.name, v.class, v.starlarkType)
1060 }
Sasha Smundak6609ba72021-07-22 18:32:56 -07001061 fs := NewFindMockFS([]string{
1062 "vendor/foo1/cfg.mk",
1063 "vendor/bar/baz/cfg.mk",
1064 "part.mk",
1065 "foo/font.mk",
1066 "bar/font.mk",
1067 })
Sasha Smundakb051c4e2020-11-05 20:45:07 -08001068 for _, test := range testCases {
1069 t.Run(test.desc,
1070 func(t *testing.T) {
1071 ss, err := Convert(Request{
1072 MkFile: test.mkname,
1073 Reader: bytes.NewBufferString(test.in),
1074 RootDir: ".",
1075 OutputSuffix: ".star",
1076 WarnPartialSuccess: true,
Sasha Smundak6609ba72021-07-22 18:32:56 -07001077 SourceFS: fs,
1078 MakefileFinder: &testMakefileFinder{fs: fs},
Sasha Smundakb051c4e2020-11-05 20:45:07 -08001079 })
1080 if err != nil {
1081 t.Error(err)
1082 return
1083 }
1084 got := ss.String()
1085 if got != test.expected {
1086 t.Errorf("%q failed\nExpected:\n%s\nActual:\n%s\n", test.desc,
1087 strings.ReplaceAll(test.expected, "\n", "␤\n"),
1088 strings.ReplaceAll(got, "\n", "␤\n"))
1089 }
1090 })
1091 }
1092}