| // Copyright 2021 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 java | 
 |  | 
 | import ( | 
 | 	"fmt" | 
 |  | 
 | 	"android/soong/android" | 
 | 	"android/soong/dexpreopt" | 
 | ) | 
 |  | 
 | func init() { | 
 | 	registerPlatformBootclasspathBuildComponents(android.InitRegistrationContext) | 
 | } | 
 |  | 
 | func registerPlatformBootclasspathBuildComponents(ctx android.RegistrationContext) { | 
 | 	ctx.RegisterParallelSingletonModuleType("platform_bootclasspath", platformBootclasspathFactory) | 
 | } | 
 |  | 
 | // The tags used for the dependencies between the platform bootclasspath and any configured boot | 
 | // jars. | 
 | var ( | 
 | 	platformBootclasspathArtBootJarDepTag  = bootclasspathDependencyTag{name: "art-boot-jar"} | 
 | 	platformBootclasspathBootJarDepTag     = bootclasspathDependencyTag{name: "platform-boot-jar"} | 
 | 	platformBootclasspathApexBootJarDepTag = bootclasspathDependencyTag{name: "apex-boot-jar"} | 
 | ) | 
 |  | 
 | type platformBootclasspathModule struct { | 
 | 	android.SingletonModuleBase | 
 | 	ClasspathFragmentBase | 
 |  | 
 | 	properties platformBootclasspathProperties | 
 |  | 
 | 	// The apex:module pairs obtained from the configured modules. | 
 | 	configuredModules []android.Module | 
 |  | 
 | 	// The apex:module pairs obtained from the fragments. | 
 | 	fragments []android.Module | 
 |  | 
 | 	// Path to the monolithic hiddenapi-flags.csv file. | 
 | 	hiddenAPIFlagsCSV android.OutputPath | 
 |  | 
 | 	// Path to the monolithic hiddenapi-index.csv file. | 
 | 	hiddenAPIIndexCSV android.OutputPath | 
 |  | 
 | 	// Path to the monolithic hiddenapi-unsupported.csv file. | 
 | 	hiddenAPIMetadataCSV android.OutputPath | 
 |  | 
 | 	// Path to a srcjar containing all the transitive sources of the bootclasspath. | 
 | 	srcjar android.OutputPath | 
 | } | 
 |  | 
 | type platformBootclasspathProperties struct { | 
 | 	BootclasspathFragmentsDepsProperties | 
 |  | 
 | 	HiddenAPIFlagFileProperties | 
 | } | 
 |  | 
 | func platformBootclasspathFactory() android.SingletonModule { | 
 | 	m := &platformBootclasspathModule{} | 
 | 	m.AddProperties(&m.properties) | 
 | 	initClasspathFragment(m, BOOTCLASSPATH) | 
 | 	android.InitAndroidArchModule(m, android.DeviceSupported, android.MultilibCommon) | 
 | 	return m | 
 | } | 
 |  | 
 | var _ android.OutputFileProducer = (*platformBootclasspathModule)(nil) | 
 |  | 
 | func (b *platformBootclasspathModule) AndroidMkEntries() (entries []android.AndroidMkEntries) { | 
 | 	entries = append(entries, android.AndroidMkEntries{ | 
 | 		Class: "FAKE", | 
 | 		// Need at least one output file in order for this to take effect. | 
 | 		OutputFile: android.OptionalPathForPath(b.hiddenAPIFlagsCSV), | 
 | 		Include:    "$(BUILD_PHONY_PACKAGE)", | 
 | 	}) | 
 | 	entries = append(entries, b.classpathFragmentBase().androidMkEntries()...) | 
 | 	return | 
 | } | 
 |  | 
 | // Make the hidden API files available from the platform-bootclasspath module. | 
 | func (b *platformBootclasspathModule) OutputFiles(tag string) (android.Paths, error) { | 
 | 	switch tag { | 
 | 	case "hiddenapi-flags.csv": | 
 | 		return android.Paths{b.hiddenAPIFlagsCSV}, nil | 
 | 	case "hiddenapi-index.csv": | 
 | 		return android.Paths{b.hiddenAPIIndexCSV}, nil | 
 | 	case "hiddenapi-metadata.csv": | 
 | 		return android.Paths{b.hiddenAPIMetadataCSV}, nil | 
 | 	case ".srcjar": | 
 | 		return android.Paths{b.srcjar}, nil | 
 | 	} | 
 |  | 
 | 	return nil, fmt.Errorf("unknown tag %s", tag) | 
 | } | 
 |  | 
 | func (b *platformBootclasspathModule) DepsMutator(ctx android.BottomUpMutatorContext) { | 
 | 	b.hiddenAPIDepsMutator(ctx) | 
 |  | 
 | 	if !dexpreopt.IsDex2oatNeeded(ctx) { | 
 | 		return | 
 | 	} | 
 |  | 
 | 	// Add a dependency onto the dex2oat tool which is needed for creating the boot image. The | 
 | 	// path is retrieved from the dependency by GetGlobalSoongConfig(ctx). | 
 | 	dexpreopt.RegisterToolDeps(ctx) | 
 | } | 
 |  | 
 | func (b *platformBootclasspathModule) hiddenAPIDepsMutator(ctx android.BottomUpMutatorContext) { | 
 | 	if ctx.Config().DisableHiddenApiChecks() { | 
 | 		return | 
 | 	} | 
 |  | 
 | 	// Add dependencies onto the stub lib modules. | 
 | 	apiLevelToStubLibModules := hiddenAPIComputeMonolithicStubLibModules(ctx.Config()) | 
 | 	hiddenAPIAddStubLibDependencies(ctx, apiLevelToStubLibModules) | 
 | } | 
 |  | 
 | func (b *platformBootclasspathModule) BootclasspathDepsMutator(ctx android.BottomUpMutatorContext) { | 
 | 	// Add dependencies on all the ART jars. | 
 | 	global := dexpreopt.GetGlobalConfig(ctx) | 
 | 	addDependenciesOntoBootImageModules(ctx, global.ArtApexJars, platformBootclasspathArtBootJarDepTag) | 
 |  | 
 | 	// Add dependencies on all the non-updatable jars, which are on the platform or in non-updatable | 
 | 	// APEXes. | 
 | 	addDependenciesOntoBootImageModules(ctx, b.platformJars(ctx), platformBootclasspathBootJarDepTag) | 
 |  | 
 | 	// Add dependencies on all the updatable jars, except the ART jars. | 
 | 	apexJars := dexpreopt.GetGlobalConfig(ctx).ApexBootJars | 
 | 	addDependenciesOntoBootImageModules(ctx, apexJars, platformBootclasspathApexBootJarDepTag) | 
 |  | 
 | 	// Add dependencies on all the fragments. | 
 | 	b.properties.BootclasspathFragmentsDepsProperties.addDependenciesOntoFragments(ctx) | 
 | } | 
 |  | 
 | func addDependenciesOntoBootImageModules(ctx android.BottomUpMutatorContext, modules android.ConfiguredJarList, tag bootclasspathDependencyTag) { | 
 | 	for i := 0; i < modules.Len(); i++ { | 
 | 		apex := modules.Apex(i) | 
 | 		name := modules.Jar(i) | 
 |  | 
 | 		addDependencyOntoApexModulePair(ctx, apex, name, tag) | 
 | 	} | 
 | } | 
 |  | 
 | // GenerateSingletonBuildActions does nothing and must never do anything. | 
 | // | 
 | // This module only implements android.SingletonModule so that it can implement | 
 | // android.SingletonMakeVarsProvider. | 
 | func (b *platformBootclasspathModule) GenerateSingletonBuildActions(android.SingletonContext) { | 
 | 	// Keep empty | 
 | } | 
 |  | 
 | func (d *platformBootclasspathModule) MakeVars(ctx android.MakeVarsContext) { | 
 | 	d.generateHiddenApiMakeVars(ctx) | 
 | } | 
 |  | 
 | func (b *platformBootclasspathModule) GenerateAndroidBuildActions(ctx android.ModuleContext) { | 
 | 	// Gather all the dependencies from the art, platform, and apex boot jars. | 
 | 	artModules := gatherApexModulePairDepsWithTag(ctx, platformBootclasspathArtBootJarDepTag) | 
 | 	platformModules := gatherApexModulePairDepsWithTag(ctx, platformBootclasspathBootJarDepTag) | 
 | 	apexModules := gatherApexModulePairDepsWithTag(ctx, platformBootclasspathApexBootJarDepTag) | 
 |  | 
 | 	// Concatenate them all, in order as they would appear on the bootclasspath. | 
 | 	var allModules []android.Module | 
 | 	allModules = append(allModules, artModules...) | 
 | 	allModules = append(allModules, platformModules...) | 
 | 	allModules = append(allModules, apexModules...) | 
 | 	b.configuredModules = allModules | 
 |  | 
 | 	var transitiveSrcFiles android.Paths | 
 | 	for _, module := range allModules { | 
 | 		depInfo := ctx.OtherModuleProvider(module, JavaInfoProvider).(JavaInfo) | 
 | 		if depInfo.TransitiveSrcFiles != nil { | 
 | 			transitiveSrcFiles = append(transitiveSrcFiles, depInfo.TransitiveSrcFiles.ToList()...) | 
 | 		} | 
 | 	} | 
 | 	jarArgs := resourcePathsToJarArgs(transitiveSrcFiles) | 
 | 	jarArgs = append(jarArgs, "-srcjar") // Move srcfiles to the right package | 
 | 	b.srcjar = android.PathForModuleOut(ctx, ctx.ModuleName()+"-transitive.srcjar").OutputPath | 
 | 	TransformResourcesToJar(ctx, b.srcjar, jarArgs, transitiveSrcFiles) | 
 |  | 
 | 	// Gather all the fragments dependencies. | 
 | 	b.fragments = gatherApexModulePairDepsWithTag(ctx, bootclasspathFragmentDepTag) | 
 |  | 
 | 	// Check the configuration of the boot modules. | 
 | 	// ART modules are checked by the art-bootclasspath-fragment. | 
 | 	b.checkPlatformModules(ctx, platformModules) | 
 | 	b.checkApexModules(ctx, apexModules) | 
 |  | 
 | 	b.generateClasspathProtoBuildActions(ctx) | 
 |  | 
 | 	bootDexJarByModule := b.generateHiddenAPIBuildActions(ctx, b.configuredModules, b.fragments) | 
 | 	buildRuleForBootJarsPackageCheck(ctx, bootDexJarByModule) | 
 | } | 
 |  | 
 | // Generate classpaths.proto config | 
 | func (b *platformBootclasspathModule) generateClasspathProtoBuildActions(ctx android.ModuleContext) { | 
 | 	configuredJars := b.configuredJars(ctx) | 
 | 	// ART and platform boot jars must have a corresponding entry in DEX2OATBOOTCLASSPATH | 
 | 	classpathJars := configuredJarListToClasspathJars(ctx, configuredJars, BOOTCLASSPATH, DEX2OATBOOTCLASSPATH) | 
 | 	b.classpathFragmentBase().generateClasspathProtoBuildActions(ctx, configuredJars, classpathJars) | 
 | } | 
 |  | 
 | func (b *platformBootclasspathModule) configuredJars(ctx android.ModuleContext) android.ConfiguredJarList { | 
 | 	// Include all non APEX jars | 
 | 	jars := b.platformJars(ctx) | 
 |  | 
 | 	// Include jars from APEXes that don't populate their classpath proto config. | 
 | 	remainingJars := dexpreopt.GetGlobalConfig(ctx).ApexBootJars | 
 | 	for _, fragment := range b.fragments { | 
 | 		info := ctx.OtherModuleProvider(fragment, ClasspathFragmentProtoContentInfoProvider).(ClasspathFragmentProtoContentInfo) | 
 | 		if info.ClasspathFragmentProtoGenerated { | 
 | 			remainingJars = remainingJars.RemoveList(info.ClasspathFragmentProtoContents) | 
 | 		} | 
 | 	} | 
 | 	for i := 0; i < remainingJars.Len(); i++ { | 
 | 		jars = jars.Append(remainingJars.Apex(i), remainingJars.Jar(i)) | 
 | 	} | 
 |  | 
 | 	return jars | 
 | } | 
 |  | 
 | func (b *platformBootclasspathModule) platformJars(ctx android.PathContext) android.ConfiguredJarList { | 
 | 	global := dexpreopt.GetGlobalConfig(ctx) | 
 | 	return global.BootJars.RemoveList(global.ArtApexJars) | 
 | } | 
 |  | 
 | // checkPlatformModules ensures that the non-updatable modules supplied are not part of an | 
 | // apex module. | 
 | func (b *platformBootclasspathModule) checkPlatformModules(ctx android.ModuleContext, modules []android.Module) { | 
 | 	// TODO(satayev): change this check to only allow core-icu4j, all apex jars should not be here. | 
 | 	for _, m := range modules { | 
 | 		apexInfo := ctx.OtherModuleProvider(m, android.ApexInfoProvider).(android.ApexInfo) | 
 | 		fromUpdatableApex := apexInfo.Updatable | 
 | 		if fromUpdatableApex { | 
 | 			// error: this jar is part of an updatable apex | 
 | 			ctx.ModuleErrorf("module %q from updatable apexes %q is not allowed in the platform bootclasspath", ctx.OtherModuleName(m), apexInfo.InApexVariants) | 
 | 		} else { | 
 | 			// ok: this jar is part of the platform or a non-updatable apex | 
 | 		} | 
 | 	} | 
 | } | 
 |  | 
 | // checkApexModules ensures that the apex modules supplied are not from the platform. | 
 | func (b *platformBootclasspathModule) checkApexModules(ctx android.ModuleContext, modules []android.Module) { | 
 | 	for _, m := range modules { | 
 | 		apexInfo := ctx.OtherModuleProvider(m, android.ApexInfoProvider).(android.ApexInfo) | 
 | 		fromUpdatableApex := apexInfo.Updatable | 
 | 		if fromUpdatableApex { | 
 | 			// ok: this jar is part of an updatable apex | 
 | 		} else { | 
 | 			name := ctx.OtherModuleName(m) | 
 | 			if apexInfo.IsForPlatform() { | 
 | 				// If AlwaysUsePrebuiltSdks() returns true then it is possible that the updatable list will | 
 | 				// include platform variants of a prebuilt module due to workarounds elsewhere. In that case | 
 | 				// do not treat this as an error. | 
 | 				// TODO(b/179354495): Always treat this as an error when migration to bootclasspath_fragment | 
 | 				//  modules is complete. | 
 | 				if !ctx.Config().AlwaysUsePrebuiltSdks() { | 
 | 					// error: this jar is part of the platform | 
 | 					ctx.ModuleErrorf("module %q from platform is not allowed in the apex boot jars list", name) | 
 | 				} | 
 | 			} else { | 
 | 				// TODO(b/177892522): Treat this as an error. | 
 | 				// Cannot do that at the moment because framework-wifi and framework-tethering are in the | 
 | 				// PRODUCT_APEX_BOOT_JARS but not marked as updatable in AOSP. | 
 | 			} | 
 | 		} | 
 | 	} | 
 | } | 
 |  | 
 | // generateHiddenAPIBuildActions generates all the hidden API related build rules. | 
 | func (b *platformBootclasspathModule) generateHiddenAPIBuildActions(ctx android.ModuleContext, modules []android.Module, fragments []android.Module) bootDexJarByModule { | 
 |  | 
 | 	// Save the paths to the monolithic files for retrieval via OutputFiles(). | 
 | 	b.hiddenAPIFlagsCSV = hiddenAPISingletonPaths(ctx).flags | 
 | 	b.hiddenAPIIndexCSV = hiddenAPISingletonPaths(ctx).index | 
 | 	b.hiddenAPIMetadataCSV = hiddenAPISingletonPaths(ctx).metadata | 
 |  | 
 | 	bootDexJarByModule := extractBootDexJarsFromModules(ctx, modules) | 
 |  | 
 | 	// Don't run any hiddenapi rules if hidden api checks are disabled. This is a performance | 
 | 	// optimization that can be used to reduce the incremental build time but as its name suggests it | 
 | 	// can be unsafe to use, e.g. when the changes affect anything that goes on the bootclasspath. | 
 | 	if ctx.Config().DisableHiddenApiChecks() { | 
 | 		paths := android.OutputPaths{b.hiddenAPIFlagsCSV, b.hiddenAPIIndexCSV, b.hiddenAPIMetadataCSV} | 
 | 		for _, path := range paths { | 
 | 			ctx.Build(pctx, android.BuildParams{ | 
 | 				Rule:   android.Touch, | 
 | 				Output: path, | 
 | 			}) | 
 | 		} | 
 | 		return bootDexJarByModule | 
 | 	} | 
 |  | 
 | 	// Construct a list of ClasspathElement objects from the modules and fragments. | 
 | 	classpathElements := CreateClasspathElements(ctx, modules, fragments) | 
 |  | 
 | 	monolithicInfo := b.createAndProvideMonolithicHiddenAPIInfo(ctx, classpathElements) | 
 |  | 
 | 	// Extract the classes jars only from those libraries that do not have corresponding fragments as | 
 | 	// the fragments will have already provided the flags that are needed. | 
 | 	classesJars := monolithicInfo.ClassesJars | 
 |  | 
 | 	// Create the input to pass to buildRuleToGenerateHiddenAPIStubFlagsFile | 
 | 	input := newHiddenAPIFlagInput() | 
 |  | 
 | 	// Gather stub library information from the dependencies on modules provided by | 
 | 	// hiddenAPIComputeMonolithicStubLibModules. | 
 | 	input.gatherStubLibInfo(ctx, nil) | 
 |  | 
 | 	// Use the flag files from this module and all the fragments. | 
 | 	input.FlagFilesByCategory = monolithicInfo.FlagsFilesByCategory | 
 |  | 
 | 	// Generate the monolithic stub-flags.csv file. | 
 | 	stubFlags := hiddenAPISingletonPaths(ctx).stubFlags | 
 | 	buildRuleToGenerateHiddenAPIStubFlagsFile(ctx, "platform-bootclasspath-monolithic-hiddenapi-stub-flags", "monolithic hidden API stub flags", stubFlags, bootDexJarByModule.bootDexJars(), input, monolithicInfo.StubFlagSubsets) | 
 |  | 
 | 	// Generate the annotation-flags.csv file from all the module annotations. | 
 | 	annotationFlags := android.PathForModuleOut(ctx, "hiddenapi-monolithic", "annotation-flags-from-classes.csv") | 
 | 	buildRuleToGenerateAnnotationFlags(ctx, "intermediate hidden API flags", classesJars, stubFlags, annotationFlags) | 
 |  | 
 | 	// Generate the monolithic hiddenapi-flags.csv file. | 
 | 	// | 
 | 	// Use annotation flags generated directly from the classes jars as well as annotation flag files | 
 | 	// provided by prebuilts. | 
 | 	allAnnotationFlagFiles := android.Paths{annotationFlags} | 
 | 	allAnnotationFlagFiles = append(allAnnotationFlagFiles, monolithicInfo.AnnotationFlagsPaths...) | 
 | 	allFlags := hiddenAPISingletonPaths(ctx).flags | 
 | 	buildRuleToGenerateHiddenApiFlags(ctx, "hiddenAPIFlagsFile", "monolithic hidden API flags", allFlags, stubFlags, allAnnotationFlagFiles, monolithicInfo.FlagsFilesByCategory, monolithicInfo.FlagSubsets, android.OptionalPath{}) | 
 |  | 
 | 	// Generate an intermediate monolithic hiddenapi-metadata.csv file directly from the annotations | 
 | 	// in the source code. | 
 | 	intermediateMetadataCSV := android.PathForModuleOut(ctx, "hiddenapi-monolithic", "metadata-from-classes.csv") | 
 | 	buildRuleToGenerateMetadata(ctx, "intermediate hidden API metadata", classesJars, stubFlags, intermediateMetadataCSV) | 
 |  | 
 | 	// Generate the monolithic hiddenapi-metadata.csv file. | 
 | 	// | 
 | 	// Use metadata files generated directly from the classes jars as well as metadata files provided | 
 | 	// by prebuilts. | 
 | 	// | 
 | 	// This has the side effect of ensuring that the output file uses | quotes just in case that is | 
 | 	// important for the tools that consume the metadata file. | 
 | 	allMetadataFlagFiles := android.Paths{intermediateMetadataCSV} | 
 | 	allMetadataFlagFiles = append(allMetadataFlagFiles, monolithicInfo.MetadataPaths...) | 
 | 	metadataCSV := hiddenAPISingletonPaths(ctx).metadata | 
 | 	b.buildRuleMergeCSV(ctx, "monolithic hidden API metadata", allMetadataFlagFiles, metadataCSV) | 
 |  | 
 | 	// Generate an intermediate monolithic hiddenapi-index.csv file directly from the CSV files in the | 
 | 	// classes jars. | 
 | 	intermediateIndexCSV := android.PathForModuleOut(ctx, "hiddenapi-monolithic", "index-from-classes.csv") | 
 | 	buildRuleToGenerateIndex(ctx, "intermediate hidden API index", classesJars, intermediateIndexCSV) | 
 |  | 
 | 	// Generate the monolithic hiddenapi-index.csv file. | 
 | 	// | 
 | 	// Use index files generated directly from the classes jars as well as index files provided | 
 | 	// by prebuilts. | 
 | 	allIndexFlagFiles := android.Paths{intermediateIndexCSV} | 
 | 	allIndexFlagFiles = append(allIndexFlagFiles, monolithicInfo.IndexPaths...) | 
 | 	indexCSV := hiddenAPISingletonPaths(ctx).index | 
 | 	b.buildRuleMergeCSV(ctx, "monolithic hidden API index", allIndexFlagFiles, indexCSV) | 
 |  | 
 | 	return bootDexJarByModule | 
 | } | 
 |  | 
 | // createAndProvideMonolithicHiddenAPIInfo creates a MonolithicHiddenAPIInfo and provides it for | 
 | // testing. | 
 | func (b *platformBootclasspathModule) createAndProvideMonolithicHiddenAPIInfo(ctx android.ModuleContext, classpathElements ClasspathElements) MonolithicHiddenAPIInfo { | 
 | 	// Create a temporary input structure in which to collate information provided directly by this | 
 | 	// module, either through properties or direct dependencies. | 
 | 	temporaryInput := newHiddenAPIFlagInput() | 
 |  | 
 | 	// Create paths to the flag files specified in the properties. | 
 | 	temporaryInput.extractFlagFilesFromProperties(ctx, &b.properties.HiddenAPIFlagFileProperties) | 
 |  | 
 | 	// Create the monolithic info, by starting with the flag files specified on this and then merging | 
 | 	// in information from all the fragment dependencies of this. | 
 | 	monolithicInfo := newMonolithicHiddenAPIInfo(ctx, temporaryInput.FlagFilesByCategory, classpathElements) | 
 |  | 
 | 	// Store the information for testing. | 
 | 	ctx.SetProvider(MonolithicHiddenAPIInfoProvider, monolithicInfo) | 
 | 	return monolithicInfo | 
 | } | 
 |  | 
 | func (b *platformBootclasspathModule) buildRuleMergeCSV(ctx android.ModuleContext, desc string, inputPaths android.Paths, outputPath android.WritablePath) { | 
 | 	rule := android.NewRuleBuilder(pctx, ctx) | 
 | 	rule.Command(). | 
 | 		BuiltTool("merge_csv"). | 
 | 		Flag("--key_field signature"). | 
 | 		FlagWithOutput("--output=", outputPath). | 
 | 		Inputs(inputPaths) | 
 |  | 
 | 	rule.Build(desc, desc) | 
 | } | 
 |  | 
 | // generateHiddenApiMakeVars generates make variables needed by hidden API related make rules, e.g. | 
 | // veridex and run-appcompat. | 
 | func (b *platformBootclasspathModule) generateHiddenApiMakeVars(ctx android.MakeVarsContext) { | 
 | 	if ctx.Config().IsEnvTrue("UNSAFE_DISABLE_HIDDENAPI_FLAGS") { | 
 | 		return | 
 | 	} | 
 | 	// INTERNAL_PLATFORM_HIDDENAPI_FLAGS is used by Make rules in art/ and cts/. | 
 | 	ctx.Strict("INTERNAL_PLATFORM_HIDDENAPI_FLAGS", b.hiddenAPIFlagsCSV.String()) | 
 | } |