| // Copyright 2020 Google Inc. All rights reserved. | 
 | // | 
 | // Licensed under the Apache License, Version 2.0 (the "License"); | 
 | // you may not use this file except in compliance with the License. | 
 | // You may obtain a copy of the License at | 
 | // | 
 | //     http://www.apache.org/licenses/LICENSE-2.0 | 
 | // | 
 | // Unless required by applicable law or agreed to in writing, software | 
 | // distributed under the License is distributed on an "AS IS" BASIS, | 
 | // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
 | // See the License for the specific language governing permissions and | 
 | // limitations under the License. | 
 |  | 
 | package cc | 
 |  | 
 | import ( | 
 | 	"android/soong/android" | 
 | 	"android/soong/genrule" | 
 | ) | 
 |  | 
 | // sdkTransitionMutator creates a platform and an SDK variant for modules | 
 | // that set sdk_version, and ignores sdk_version for the platform | 
 | // variant.  The SDK variant will be used for embedding in APKs | 
 | // that may be installed on older platforms.  Apexes use their own | 
 | // variants that enforce backwards compatibility. | 
 | type sdkTransitionMutator struct{} | 
 |  | 
 | func (sdkTransitionMutator) Split(ctx android.BaseModuleContext) []string { | 
 | 	if ctx.Os() != android.Android { | 
 | 		return []string{""} | 
 | 	} | 
 |  | 
 | 	switch m := ctx.Module().(type) { | 
 | 	case LinkableInterface: | 
 | 		if m.AlwaysSdk() { | 
 | 			if !m.UseSdk() && !m.SplitPerApiLevel() { | 
 | 				ctx.ModuleErrorf("UseSdk() must return true when AlwaysSdk is set, did the factory forget to set Sdk_version?") | 
 | 			} | 
 | 			return []string{"sdk"} | 
 | 		} else if m.UseSdk() || m.SplitPerApiLevel() { | 
 | 			return []string{"", "sdk"} | 
 | 		} else { | 
 | 			return []string{""} | 
 | 		} | 
 | 	case *genrule.Module: | 
 | 		if p, ok := m.Extra.(*GenruleExtraProperties); ok { | 
 | 			if String(p.Sdk_version) != "" { | 
 | 				return []string{"", "sdk"} | 
 | 			} else { | 
 | 				return []string{""} | 
 | 			} | 
 | 		} | 
 | 	} | 
 |  | 
 | 	return []string{""} | 
 | } | 
 |  | 
 | func (sdkTransitionMutator) OutgoingTransition(ctx android.OutgoingTransitionContext, sourceVariation string) string { | 
 | 	return sourceVariation | 
 | } | 
 |  | 
 | func (sdkTransitionMutator) IncomingTransition(ctx android.IncomingTransitionContext, incomingVariation string) string { | 
 | 	if ctx.Os() != android.Android { | 
 | 		return "" | 
 | 	} | 
 | 	switch m := ctx.Module().(type) { | 
 | 	case LinkableInterface: | 
 | 		if m.AlwaysSdk() { | 
 | 			return "sdk" | 
 | 		} else if m.UseSdk() || m.SplitPerApiLevel() { | 
 | 			return incomingVariation | 
 | 		} | 
 | 	case *genrule.Module: | 
 | 		if p, ok := m.Extra.(*GenruleExtraProperties); ok { | 
 | 			if String(p.Sdk_version) != "" { | 
 | 				return incomingVariation | 
 | 			} | 
 | 		} | 
 | 	} | 
 |  | 
 | 	if ctx.IsAddingDependency() { | 
 | 		return incomingVariation | 
 | 	} else { | 
 | 		return "" | 
 | 	} | 
 | } | 
 |  | 
 | func (sdkTransitionMutator) Mutate(ctx android.BottomUpMutatorContext, variation string) { | 
 | 	if ctx.Os() != android.Android { | 
 | 		return | 
 | 	} | 
 |  | 
 | 	switch m := ctx.Module().(type) { | 
 | 	case LinkableInterface: | 
 | 		ccModule, isCcModule := ctx.Module().(*Module) | 
 | 		if m.AlwaysSdk() { | 
 | 			if variation != "sdk" { | 
 | 				ctx.ModuleErrorf("tried to create variation %q for module with AlwaysSdk set, expected \"sdk\"", variation) | 
 | 			} | 
 |  | 
 | 			ccModule.Properties.IsSdkVariant = true | 
 | 		} else if m.UseSdk() || m.SplitPerApiLevel() { | 
 | 			if variation == "" { | 
 | 				// Clear the sdk_version property for the platform (non-SDK) variant so later code | 
 | 				// doesn't get confused by it. | 
 | 				ccModule.Properties.Sdk_version = nil | 
 | 			} else { | 
 | 				// Mark the SDK variant. | 
 | 				ccModule.Properties.IsSdkVariant = true | 
 |  | 
 | 				// SDK variant never gets installed because the variant is to be embedded in | 
 | 				// APKs, not to be installed to the platform. | 
 | 				ccModule.Properties.PreventInstall = true | 
 | 			} | 
 |  | 
 | 			if ctx.Config().UnbundledBuildApps() { | 
 | 				if variation == "" { | 
 | 					// For an unbundled apps build, hide the platform variant from Make | 
 | 					// so that other Make modules don't link against it, but against the | 
 | 					// SDK variant. | 
 | 					ccModule.Properties.HideFromMake = true | 
 | 				} | 
 | 			} else { | 
 | 				if variation == "sdk" { | 
 | 					// For a platform build, mark the SDK variant so that it gets a ".sdk" suffix when | 
 | 					// exposed to Make. | 
 | 					ccModule.Properties.SdkAndPlatformVariantVisibleToMake = true | 
 | 				} | 
 | 			} | 
 | 		} else { | 
 | 			if isCcModule { | 
 | 				// Clear the sdk_version property for modules that don't have an SDK variant so | 
 | 				// later code doesn't get confused by it. | 
 | 				ccModule.Properties.Sdk_version = nil | 
 | 			} | 
 | 		} | 
 | 	} | 
 | } |