Delete transition variation when no matching variant was found

TransitionMutators sometimes apply to one variant of a module but
not to another.  Deleting the variation when all variants were
untouched is insufficient if a post-mutator dependency is being added
on a variant that was untouched.  Delete the variation whenever
no matching incoming variant was found.

Bug: 319288033
Test: TestPostTransitionDeps
Flag: NONE
Change-Id: Ib073f6ec3090d09e4798b6f9ca3061ec5d58d722
This commit is contained in:
Colin Cross 2024-05-07 14:14:04 -07:00
parent 38b56e136f
commit 02aa24823e
2 changed files with 35 additions and 12 deletions

View file

@ -1878,14 +1878,7 @@ func (c *Context) findReverseDependency(module *moduleInfo, config any, destName
// use as the module context for IncomingTransition. // use as the module context for IncomingTransition.
func (c *Context) applyIncomingTransitions(config any, group *moduleGroup, variant variationMap, requestedVariations []Variation) { func (c *Context) applyIncomingTransitions(config any, group *moduleGroup, variant variationMap, requestedVariations []Variation) {
for _, transitionMutator := range c.transitionMutators { for _, transitionMutator := range c.transitionMutators {
if len(transitionMutator.inputVariants[group]) == 0 { appliedIncomingTransition := false
// The transition mutator didn't apply anything to the target module, remove the variation unless it
// was explicitly requested when adding the dependency.
if !slices.ContainsFunc(requestedVariations, func(v Variation) bool { return v.Mutator == transitionMutator.name }) {
delete(variant, transitionMutator.name)
}
continue
}
for _, inputVariant := range transitionMutator.inputVariants[group] { for _, inputVariant := range transitionMutator.inputVariants[group] {
if inputVariant.variant.variations.subsetOf(variant) { if inputVariant.variant.variations.subsetOf(variant) {
sourceVariation := variant[transitionMutator.name] sourceVariation := variant[transitionMutator.name]
@ -1897,9 +1890,17 @@ func (c *Context) applyIncomingTransitions(config any, group *moduleGroup, varia
outgoingVariation := transitionMutator.mutator.IncomingTransition(ctx, sourceVariation) outgoingVariation := transitionMutator.mutator.IncomingTransition(ctx, sourceVariation)
variant[transitionMutator.name] = outgoingVariation variant[transitionMutator.name] = outgoingVariation
appliedIncomingTransition = true
break break
} }
} }
if !appliedIncomingTransition {
// The transition mutator didn't apply anything to the target module, remove the variation unless it
// was explicitly requested when adding the dependency.
if !slices.ContainsFunc(requestedVariations, func(v Variation) bool { return v.Mutator == transitionMutator.name }) {
delete(variant, transitionMutator.name)
}
}
} }
} }

View file

@ -84,6 +84,10 @@ const testTransitionBp = `
transition_module { transition_module {
name: "E", name: "E",
} }
transition_module {
name: "F"
}
` `
func getTransitionModule(ctx *Context, name, variant string) *transitionModule { func getTransitionModule(ctx *Context, name, variant string) *transitionModule {
@ -137,6 +141,8 @@ func TestTransition(t *testing.T) {
checkTransitionVariants(t, ctx, "D", []string{"", "d"}) checkTransitionVariants(t, ctx, "D", []string{"", "d"})
// Module E inherits d from D // Module E inherits d from D
checkTransitionVariants(t, ctx, "E", []string{"", "d"}) checkTransitionVariants(t, ctx, "E", []string{"", "d"})
// Module F is untouched
checkTransitionVariants(t, ctx, "F", []string{""})
A_a := getTransitionModule(ctx, "A", "a") A_a := getTransitionModule(ctx, "A", "a")
A_b := getTransitionModule(ctx, "A", "b") A_b := getTransitionModule(ctx, "A", "b")
@ -147,6 +153,7 @@ func TestTransition(t *testing.T) {
C_c := getTransitionModule(ctx, "C", "c") C_c := getTransitionModule(ctx, "C", "c")
D_d := getTransitionModule(ctx, "D", "d") D_d := getTransitionModule(ctx, "D", "d")
E_d := getTransitionModule(ctx, "E", "d") E_d := getTransitionModule(ctx, "E", "d")
F := getTransitionModule(ctx, "F", "")
checkTransitionDeps(t, ctx, A_a, "B(a)", "C(a)") checkTransitionDeps(t, ctx, A_a, "B(a)", "C(a)")
checkTransitionDeps(t, ctx, A_b, "B(b)", "C(b)") checkTransitionDeps(t, ctx, A_b, "B(b)", "C(b)")
@ -157,6 +164,7 @@ func TestTransition(t *testing.T) {
checkTransitionDeps(t, ctx, C_c, "D(d)") checkTransitionDeps(t, ctx, C_c, "D(d)")
checkTransitionDeps(t, ctx, D_d, "E(d)") checkTransitionDeps(t, ctx, D_d, "E(d)")
checkTransitionDeps(t, ctx, E_d) checkTransitionDeps(t, ctx, E_d)
checkTransitionDeps(t, ctx, F)
checkTransitionMutate(t, A_a, "a") checkTransitionMutate(t, A_a, "a")
checkTransitionMutate(t, A_b, "b") checkTransitionMutate(t, A_b, "b")
@ -167,11 +175,12 @@ func TestTransition(t *testing.T) {
checkTransitionMutate(t, C_c, "c") checkTransitionMutate(t, C_c, "c")
checkTransitionMutate(t, D_d, "d") checkTransitionMutate(t, D_d, "d")
checkTransitionMutate(t, E_d, "d") checkTransitionMutate(t, E_d, "d")
checkTransitionMutate(t, F, "")
} }
func TestPostTransitionDeps(t *testing.T) { func TestPostTransitionDeps(t *testing.T) {
ctx, errs := testTransition(fmt.Sprintf(testTransitionBp, ctx, errs := testTransition(fmt.Sprintf(testTransitionBp,
`post_transition_deps: ["D:late", "E:d"],`)) `post_transition_deps: ["D:late", "E:d", "F"],`))
assertNoErrors(t, errs) assertNoErrors(t, errs)
// Module A uses Split to create a and b variants // Module A uses Split to create a and b variants
@ -184,6 +193,8 @@ func TestPostTransitionDeps(t *testing.T) {
checkTransitionVariants(t, ctx, "D", []string{"", "d"}) checkTransitionVariants(t, ctx, "D", []string{"", "d"})
// Module E inherits d from D // Module E inherits d from D
checkTransitionVariants(t, ctx, "E", []string{"", "d"}) checkTransitionVariants(t, ctx, "E", []string{"", "d"})
// Module F is untouched
checkTransitionVariants(t, ctx, "F", []string{""})
A_a := getTransitionModule(ctx, "A", "a") A_a := getTransitionModule(ctx, "A", "a")
A_b := getTransitionModule(ctx, "A", "b") A_b := getTransitionModule(ctx, "A", "b")
@ -194,16 +205,23 @@ func TestPostTransitionDeps(t *testing.T) {
C_c := getTransitionModule(ctx, "C", "c") C_c := getTransitionModule(ctx, "C", "c")
D_d := getTransitionModule(ctx, "D", "d") D_d := getTransitionModule(ctx, "D", "d")
E_d := getTransitionModule(ctx, "E", "d") E_d := getTransitionModule(ctx, "E", "d")
F := getTransitionModule(ctx, "F", "")
checkTransitionDeps(t, ctx, A_a, "B(a)", "C(a)") checkTransitionDeps(t, ctx, A_a, "B(a)", "C(a)")
checkTransitionDeps(t, ctx, A_b, "B(b)", "C(b)") checkTransitionDeps(t, ctx, A_b, "B(b)", "C(b)")
checkTransitionDeps(t, ctx, B_a, "C(c)", "D(d)", "E(d)") // Verify post-mutator dependencies added to B:
checkTransitionDeps(t, ctx, B_b, "C(c)", "D(d)", "E(d)") // C(c) is a pre-mutator dependency
// D(d) was added by D:late and rewritten by IncomingTransition on D
// E(d) was added by E:d
// F() was added by F, and ignored the existing variation on B
checkTransitionDeps(t, ctx, B_a, "C(c)", "D(d)", "E(d)", "F()")
checkTransitionDeps(t, ctx, B_b, "C(c)", "D(d)", "E(d)", "F()")
checkTransitionDeps(t, ctx, C_a, "D(d)") checkTransitionDeps(t, ctx, C_a, "D(d)")
checkTransitionDeps(t, ctx, C_b, "D(d)") checkTransitionDeps(t, ctx, C_b, "D(d)")
checkTransitionDeps(t, ctx, C_c, "D(d)") checkTransitionDeps(t, ctx, C_c, "D(d)")
checkTransitionDeps(t, ctx, D_d, "E(d)") checkTransitionDeps(t, ctx, D_d, "E(d)")
checkTransitionDeps(t, ctx, E_d) checkTransitionDeps(t, ctx, E_d)
checkTransitionDeps(t, ctx, F)
checkTransitionMutate(t, A_a, "a") checkTransitionMutate(t, A_a, "a")
checkTransitionMutate(t, A_b, "b") checkTransitionMutate(t, A_b, "b")
@ -214,6 +232,7 @@ func TestPostTransitionDeps(t *testing.T) {
checkTransitionMutate(t, C_c, "c") checkTransitionMutate(t, C_c, "c")
checkTransitionMutate(t, D_d, "d") checkTransitionMutate(t, D_d, "d")
checkTransitionMutate(t, E_d, "d") checkTransitionMutate(t, E_d, "d")
checkTransitionMutate(t, F, "")
} }
func TestPostTransitionDepsMissingVariant(t *testing.T) { func TestPostTransitionDepsMissingVariant(t *testing.T) {
@ -290,7 +309,10 @@ func postTransitionDepsMutator(mctx BottomUpMutatorContext) {
if m, ok := mctx.Module().(*transitionModule); ok { if m, ok := mctx.Module().(*transitionModule); ok {
for _, dep := range m.properties.Post_transition_deps { for _, dep := range m.properties.Post_transition_deps {
module, variation, _ := strings.Cut(dep, ":") module, variation, _ := strings.Cut(dep, ":")
variations := []Variation{{"transition", variation}} var variations []Variation
if variation != "" {
variations = append(variations, Variation{"transition", variation})
}
mctx.AddVariationDependencies(variations, walkerDepsTag{follow: true}, module) mctx.AddVariationDependencies(variations, walkerDepsTag{follow: true}, module)
} }
} }