blob: 5bc3b83542d21c46c2a2b3d254de184627a2c6a8 [file] [log] [blame]
Jiyong Park9b409bc2019-10-11 14:59:13 +09001// Copyright (C) 2019 The Android Open Source Project
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 sdk
16
17import (
18 "fmt"
Paul Duffinb645ec82019-11-27 17:43:54 +000019 "reflect"
Jiyong Park9b409bc2019-10-11 14:59:13 +090020 "strings"
21
Paul Duffin375058f2019-11-29 20:17:53 +000022 "github.com/google/blueprint"
Jiyong Park9b409bc2019-10-11 14:59:13 +090023 "github.com/google/blueprint/proptools"
24
25 "android/soong/android"
Jiyong Park9b409bc2019-10-11 14:59:13 +090026)
27
28var pctx = android.NewPackageContext("android/soong/sdk")
29
Paul Duffin375058f2019-11-29 20:17:53 +000030var (
31 repackageZip = pctx.AndroidStaticRule("SnapshotRepackageZip",
32 blueprint.RuleParams{
Paul Duffince482dc2019-12-09 19:58:17 +000033 Command: `${config.Zip2ZipCmd} -i $in -o $out -x META-INF/**/* "**/*:$destdir"`,
Paul Duffin375058f2019-11-29 20:17:53 +000034 CommandDeps: []string{
35 "${config.Zip2ZipCmd}",
36 },
37 },
38 "destdir")
39
40 zipFiles = pctx.AndroidStaticRule("SnapshotZipFiles",
41 blueprint.RuleParams{
42 Command: `${config.SoongZipCmd} -C $basedir -l $out.rsp -o $out`,
43 CommandDeps: []string{
44 "${config.SoongZipCmd}",
45 },
46 Rspfile: "$out.rsp",
47 RspfileContent: "$in",
48 },
49 "basedir")
50
51 mergeZips = pctx.AndroidStaticRule("SnapshotMergeZips",
52 blueprint.RuleParams{
53 Command: `${config.MergeZipsCmd} $out $in`,
54 CommandDeps: []string{
55 "${config.MergeZipsCmd}",
56 },
57 })
58)
59
Paul Duffinb645ec82019-11-27 17:43:54 +000060type generatedContents struct {
Jiyong Park73c54ee2019-10-22 20:31:18 +090061 content strings.Builder
62 indentLevel int
Jiyong Park9b409bc2019-10-11 14:59:13 +090063}
64
Paul Duffinb645ec82019-11-27 17:43:54 +000065// generatedFile abstracts operations for writing contents into a file and emit a build rule
66// for the file.
67type generatedFile struct {
68 generatedContents
69 path android.OutputPath
70}
71
Jiyong Park232e7852019-11-04 12:23:40 +090072func newGeneratedFile(ctx android.ModuleContext, path ...string) *generatedFile {
Jiyong Park9b409bc2019-10-11 14:59:13 +090073 return &generatedFile{
Paul Duffinb645ec82019-11-27 17:43:54 +000074 path: android.PathForModuleOut(ctx, path...).OutputPath,
Jiyong Park9b409bc2019-10-11 14:59:13 +090075 }
76}
77
Paul Duffinb645ec82019-11-27 17:43:54 +000078func (gc *generatedContents) Indent() {
79 gc.indentLevel++
Jiyong Park73c54ee2019-10-22 20:31:18 +090080}
81
Paul Duffinb645ec82019-11-27 17:43:54 +000082func (gc *generatedContents) Dedent() {
83 gc.indentLevel--
Jiyong Park73c54ee2019-10-22 20:31:18 +090084}
85
Paul Duffinb645ec82019-11-27 17:43:54 +000086func (gc *generatedContents) Printfln(format string, args ...interface{}) {
Jiyong Park9b409bc2019-10-11 14:59:13 +090087 // ninja consumes newline characters in rspfile_content. Prevent it by
Paul Duffin0e0cf1d2019-11-12 19:39:25 +000088 // escaping the backslash in the newline character. The extra backslash
Jiyong Park9b409bc2019-10-11 14:59:13 +090089 // is removed when the rspfile is written to the actual script file
Paul Duffinb645ec82019-11-27 17:43:54 +000090 fmt.Fprintf(&(gc.content), strings.Repeat(" ", gc.indentLevel)+format+"\\n", args...)
Jiyong Park9b409bc2019-10-11 14:59:13 +090091}
92
93func (gf *generatedFile) build(pctx android.PackageContext, ctx android.BuilderContext, implicits android.Paths) {
94 rb := android.NewRuleBuilder()
95 // convert \\n to \n
96 rb.Command().
97 Implicits(implicits).
98 Text("echo").Text(proptools.ShellEscape(gf.content.String())).
99 Text("| sed 's/\\\\n/\\n/g' >").Output(gf.path)
100 rb.Command().
101 Text("chmod a+x").Output(gf.path)
102 rb.Build(pctx, ctx, gf.path.Base(), "Build "+gf.path.Base())
103}
104
Paul Duffin13879572019-11-28 14:31:38 +0000105// Collect all the members.
106//
107// The members are first grouped by type and then grouped by name. The order of
Paul Duffin255f18e2019-12-13 11:22:16 +0000108// the types is the order they are referenced in android.SdkMemberTypes. The
Paul Duffin13879572019-11-28 14:31:38 +0000109// names are in order in which the dependencies were added.
Paul Duffin255f18e2019-12-13 11:22:16 +0000110func (s *sdk) collectMembers(ctx android.ModuleContext) []*sdkMember {
Paul Duffin13879572019-11-28 14:31:38 +0000111 byType := make(map[android.SdkMemberType][]*sdkMember)
112 byName := make(map[string]*sdkMember)
113
Jiyong Park9b409bc2019-10-11 14:59:13 +0900114 ctx.VisitDirectDeps(func(m android.Module) {
Paul Duffin13879572019-11-28 14:31:38 +0000115 tag := ctx.OtherModuleDependencyTag(m)
116 if memberTag, ok := tag.(*sdkMemberDependencyTag); ok {
Paul Duffin255f18e2019-12-13 11:22:16 +0000117 memberType := memberTag.memberType
Jiyong Park9b409bc2019-10-11 14:59:13 +0900118
Paul Duffin13879572019-11-28 14:31:38 +0000119 // Make sure that the resolved module is allowed in the member list property.
120 if !memberType.IsInstance(m) {
Paul Duffin255f18e2019-12-13 11:22:16 +0000121 ctx.ModuleErrorf("module %q is not valid in property %s", ctx.OtherModuleName(m), memberType.SdkPropertyName())
Jiyong Park73c54ee2019-10-22 20:31:18 +0900122 }
Paul Duffin13879572019-11-28 14:31:38 +0000123
124 name := ctx.OtherModuleName(m)
125
126 member := byName[name]
127 if member == nil {
128 member = &sdkMember{memberType: memberType, name: name}
129 byName[name] = member
130 byType[memberType] = append(byType[memberType], member)
Jiyong Park73c54ee2019-10-22 20:31:18 +0900131 }
Paul Duffin13879572019-11-28 14:31:38 +0000132
133 member.variants = append(member.variants, m.(android.SdkAware))
Jiyong Park73c54ee2019-10-22 20:31:18 +0900134 }
Paul Duffin13879572019-11-28 14:31:38 +0000135 })
136
137 var members []*sdkMember
Paul Duffin255f18e2019-12-13 11:22:16 +0000138 for _, memberListProperty := range s.dynamicSdkMemberTypes.memberListProperties {
Paul Duffin13879572019-11-28 14:31:38 +0000139 membersOfType := byType[memberListProperty.memberType]
140 members = append(members, membersOfType...)
Jiyong Park9b409bc2019-10-11 14:59:13 +0900141 }
142
Paul Duffin13879572019-11-28 14:31:38 +0000143 return members
Jiyong Park73c54ee2019-10-22 20:31:18 +0900144}
Jiyong Park9b409bc2019-10-11 14:59:13 +0900145
Jiyong Park73c54ee2019-10-22 20:31:18 +0900146// SDK directory structure
147// <sdk_root>/
148// Android.bp : definition of a 'sdk' module is here. This is a hand-made one.
149// <api_ver>/ : below this directory are all auto-generated
150// Android.bp : definition of 'sdk_snapshot' module is here
151// aidl/
152// frameworks/base/core/..../IFoo.aidl : an exported AIDL file
153// java/
Jiyong Park232e7852019-11-04 12:23:40 +0900154// <module_name>.jar : the stub jar for a java library 'module_name'
Jiyong Park73c54ee2019-10-22 20:31:18 +0900155// include/
156// bionic/libc/include/stdlib.h : an exported header file
157// include_gen/
Jiyong Park232e7852019-11-04 12:23:40 +0900158// <module_name>/com/android/.../IFoo.h : a generated header file
Jiyong Park73c54ee2019-10-22 20:31:18 +0900159// <arch>/include/ : arch-specific exported headers
160// <arch>/include_gen/ : arch-specific generated headers
161// <arch>/lib/
162// libFoo.so : a stub library
163
Jiyong Park232e7852019-11-04 12:23:40 +0900164// A name that uniquely identifies a prebuilt SDK member for a version of SDK snapshot
Jiyong Park73c54ee2019-10-22 20:31:18 +0900165// This isn't visible to users, so could be changed in future.
166func versionedSdkMemberName(ctx android.ModuleContext, memberName string, version string) string {
167 return ctx.ModuleName() + "_" + memberName + string(android.SdkVersionSeparator) + version
168}
169
Jiyong Park232e7852019-11-04 12:23:40 +0900170// buildSnapshot is the main function in this source file. It creates rules to copy
171// the contents (header files, stub libraries, etc) into the zip file.
172func (s *sdk) buildSnapshot(ctx android.ModuleContext) android.OutputPath {
Paul Duffin0e0cf1d2019-11-12 19:39:25 +0000173 snapshotDir := android.PathForModuleOut(ctx, "snapshot")
Jiyong Park9b409bc2019-10-11 14:59:13 +0900174
Paul Duffin0e0cf1d2019-11-12 19:39:25 +0000175 bp := newGeneratedFile(ctx, "snapshot", "Android.bp")
Paul Duffinb645ec82019-11-27 17:43:54 +0000176
177 bpFile := &bpFile{
178 modules: make(map[string]*bpModule),
179 }
Paul Duffin0e0cf1d2019-11-12 19:39:25 +0000180
181 builder := &snapshotBuilder{
Paul Duffinb645ec82019-11-27 17:43:54 +0000182 ctx: ctx,
Paul Duffine44358f2019-11-26 18:04:12 +0000183 sdk: s,
Paul Duffinb645ec82019-11-27 17:43:54 +0000184 version: "current",
185 snapshotDir: snapshotDir.OutputPath,
Paul Duffinc62a5102019-12-11 18:34:15 +0000186 copies: make(map[string]string),
Paul Duffinb645ec82019-11-27 17:43:54 +0000187 filesToZip: []android.Path{bp.path},
188 bpFile: bpFile,
189 prebuiltModules: make(map[string]*bpModule),
Jiyong Park73c54ee2019-10-22 20:31:18 +0900190 }
Paul Duffinac37c502019-11-26 18:02:20 +0000191 s.builderForTests = builder
Jiyong Park9b409bc2019-10-11 14:59:13 +0900192
Paul Duffin255f18e2019-12-13 11:22:16 +0000193 for _, member := range s.collectMembers(ctx) {
Paul Duffin13879572019-11-28 14:31:38 +0000194 member.memberType.BuildSnapshot(ctx, builder, member)
Jiyong Park73c54ee2019-10-22 20:31:18 +0900195 }
Jiyong Park9b409bc2019-10-11 14:59:13 +0900196
Paul Duffinb645ec82019-11-27 17:43:54 +0000197 for _, unversioned := range builder.prebuiltOrder {
198 // Copy the unversioned module so it can be modified to make it versioned.
199 versioned := unversioned.copy()
200 name := versioned.properties["name"].(string)
201 versioned.setProperty("name", builder.versionedSdkMemberName(name))
202 versioned.insertAfter("name", "sdk_member_name", name)
203 bpFile.AddModule(versioned)
Paul Duffin0e0cf1d2019-11-12 19:39:25 +0000204
Paul Duffinb645ec82019-11-27 17:43:54 +0000205 // Set prefer: false - this is not strictly required as that is the default.
206 unversioned.insertAfter("name", "prefer", false)
207 bpFile.AddModule(unversioned)
208 }
209
210 // Create the snapshot module.
211 snapshotName := ctx.ModuleName() + string(android.SdkVersionSeparator) + builder.version
Paul Duffin8150da62019-12-16 17:21:27 +0000212 var snapshotModuleType string
213 if s.properties.Module_exports {
214 snapshotModuleType = "module_exports_snapshot"
215 } else {
216 snapshotModuleType = "sdk_snapshot"
217 }
218 snapshotModule := bpFile.newModule(snapshotModuleType)
Paul Duffinb645ec82019-11-27 17:43:54 +0000219 snapshotModule.AddProperty("name", snapshotName)
Paul Duffin593b3c92019-12-05 14:31:48 +0000220
221 // Make sure that the snapshot has the same visibility as the sdk.
222 visibility := android.EffectiveVisibilityRules(ctx, s)
223 if len(visibility) != 0 {
224 snapshotModule.AddProperty("visibility", visibility)
225 }
226
Paul Duffine44358f2019-11-26 18:04:12 +0000227 addHostDeviceSupportedProperties(&s.ModuleBase, snapshotModule)
Paul Duffin255f18e2019-12-13 11:22:16 +0000228 for _, memberListProperty := range s.dynamicSdkMemberTypes.memberListProperties {
229 names := memberListProperty.getter(s.dynamicMemberTypeListProperties)
Paul Duffin13879572019-11-28 14:31:38 +0000230 if len(names) > 0 {
Paul Duffin255f18e2019-12-13 11:22:16 +0000231 snapshotModule.AddProperty(memberListProperty.propertyName(), builder.versionedSdkMemberNames(names))
Paul Duffin13879572019-11-28 14:31:38 +0000232 }
Paul Duffin0e0cf1d2019-11-12 19:39:25 +0000233 }
Paul Duffinb645ec82019-11-27 17:43:54 +0000234 bpFile.AddModule(snapshotModule)
235
236 // generate Android.bp
237 bp = newGeneratedFile(ctx, "snapshot", "Android.bp")
238 generateBpContents(&bp.generatedContents, bpFile)
Paul Duffin0e0cf1d2019-11-12 19:39:25 +0000239
240 bp.build(pctx, ctx, nil)
241
242 filesToZip := builder.filesToZip
Jiyong Park9b409bc2019-10-11 14:59:13 +0900243
Jiyong Park232e7852019-11-04 12:23:40 +0900244 // zip them all
Paul Duffin91547182019-11-12 19:39:36 +0000245 outputZipFile := android.PathForModuleOut(ctx, ctx.ModuleName()+"-current.zip").OutputPath
Paul Duffin91547182019-11-12 19:39:36 +0000246 outputDesc := "Building snapshot for " + ctx.ModuleName()
247
248 // If there are no zips to merge then generate the output zip directly.
249 // Otherwise, generate an intermediate zip file into which other zips can be
250 // merged.
251 var zipFile android.OutputPath
Paul Duffin91547182019-11-12 19:39:36 +0000252 var desc string
253 if len(builder.zipsToMerge) == 0 {
254 zipFile = outputZipFile
Paul Duffin91547182019-11-12 19:39:36 +0000255 desc = outputDesc
256 } else {
257 zipFile = android.PathForModuleOut(ctx, ctx.ModuleName()+"-current.unmerged.zip").OutputPath
Paul Duffin91547182019-11-12 19:39:36 +0000258 desc = "Building intermediate snapshot for " + ctx.ModuleName()
259 }
260
Paul Duffin375058f2019-11-29 20:17:53 +0000261 ctx.Build(pctx, android.BuildParams{
262 Description: desc,
263 Rule: zipFiles,
264 Inputs: filesToZip,
265 Output: zipFile,
266 Args: map[string]string{
267 "basedir": builder.snapshotDir.String(),
268 },
269 })
Jiyong Park9b409bc2019-10-11 14:59:13 +0900270
Paul Duffin91547182019-11-12 19:39:36 +0000271 if len(builder.zipsToMerge) != 0 {
Paul Duffin375058f2019-11-29 20:17:53 +0000272 ctx.Build(pctx, android.BuildParams{
273 Description: outputDesc,
274 Rule: mergeZips,
275 Input: zipFile,
276 Inputs: builder.zipsToMerge,
277 Output: outputZipFile,
278 })
Paul Duffin91547182019-11-12 19:39:36 +0000279 }
280
281 return outputZipFile
Jiyong Park9b409bc2019-10-11 14:59:13 +0900282}
Paul Duffin0e0cf1d2019-11-12 19:39:25 +0000283
Paul Duffinb645ec82019-11-27 17:43:54 +0000284func generateBpContents(contents *generatedContents, bpFile *bpFile) {
285 contents.Printfln("// This is auto-generated. DO NOT EDIT.")
286 for _, bpModule := range bpFile.order {
287 contents.Printfln("")
288 contents.Printfln("%s {", bpModule.moduleType)
289 outputPropertySet(contents, &bpModule.bpPropertySet)
290 contents.Printfln("}")
291 }
Paul Duffinb645ec82019-11-27 17:43:54 +0000292}
293
294func outputPropertySet(contents *generatedContents, set *bpPropertySet) {
295 contents.Indent()
296 for _, name := range set.order {
297 value := set.properties[name]
298
299 reflectedValue := reflect.ValueOf(value)
300 t := reflectedValue.Type()
301
302 kind := t.Kind()
303 switch kind {
304 case reflect.Slice:
305 length := reflectedValue.Len()
306 if length > 1 {
307 contents.Printfln("%s: [", name)
308 contents.Indent()
309 for i := 0; i < length; i = i + 1 {
310 contents.Printfln("%q,", reflectedValue.Index(i).Interface())
311 }
312 contents.Dedent()
313 contents.Printfln("],")
314 } else if length == 0 {
315 contents.Printfln("%s: [],", name)
316 } else {
317 contents.Printfln("%s: [%q],", name, reflectedValue.Index(0).Interface())
318 }
319 case reflect.Bool:
320 contents.Printfln("%s: %t,", name, reflectedValue.Bool())
321
322 case reflect.Ptr:
323 contents.Printfln("%s: {", name)
324 outputPropertySet(contents, reflectedValue.Interface().(*bpPropertySet))
325 contents.Printfln("},")
326
327 default:
328 contents.Printfln("%s: %q,", name, value)
329 }
330 }
331 contents.Dedent()
332}
333
Paul Duffinac37c502019-11-26 18:02:20 +0000334func (s *sdk) GetAndroidBpContentsForTests() string {
Paul Duffinb645ec82019-11-27 17:43:54 +0000335 contents := &generatedContents{}
336 generateBpContents(contents, s.builderForTests.bpFile)
337 return contents.content.String()
Paul Duffinac37c502019-11-26 18:02:20 +0000338}
339
Paul Duffin0e0cf1d2019-11-12 19:39:25 +0000340type snapshotBuilder struct {
Paul Duffinb645ec82019-11-27 17:43:54 +0000341 ctx android.ModuleContext
Paul Duffine44358f2019-11-26 18:04:12 +0000342 sdk *sdk
Paul Duffinb645ec82019-11-27 17:43:54 +0000343 version string
344 snapshotDir android.OutputPath
345 bpFile *bpFile
Paul Duffinc62a5102019-12-11 18:34:15 +0000346
347 // Map from destination to source of each copy - used to eliminate duplicates and
348 // detect conflicts.
349 copies map[string]string
350
Paul Duffinb645ec82019-11-27 17:43:54 +0000351 filesToZip android.Paths
352 zipsToMerge android.Paths
353
354 prebuiltModules map[string]*bpModule
355 prebuiltOrder []*bpModule
Paul Duffin0e0cf1d2019-11-12 19:39:25 +0000356}
357
358func (s *snapshotBuilder) CopyToSnapshot(src android.Path, dest string) {
Paul Duffinc62a5102019-12-11 18:34:15 +0000359 if existing, ok := s.copies[dest]; ok {
360 if existing != src.String() {
361 s.ctx.ModuleErrorf("conflicting copy, %s copied from both %s and %s", dest, existing, src)
362 return
363 }
364 } else {
365 path := s.snapshotDir.Join(s.ctx, dest)
366 s.ctx.Build(pctx, android.BuildParams{
367 Rule: android.Cp,
368 Input: src,
369 Output: path,
370 })
371 s.filesToZip = append(s.filesToZip, path)
372
373 s.copies[dest] = src.String()
374 }
Paul Duffin0e0cf1d2019-11-12 19:39:25 +0000375}
376
Paul Duffin91547182019-11-12 19:39:36 +0000377func (s *snapshotBuilder) UnzipToSnapshot(zipPath android.Path, destDir string) {
378 ctx := s.ctx
379
380 // Repackage the zip file so that the entries are in the destDir directory.
381 // This will allow the zip file to be merged into the snapshot.
382 tmpZipPath := android.PathForModuleOut(ctx, "tmp", destDir+".zip").OutputPath
Paul Duffin375058f2019-11-29 20:17:53 +0000383
384 ctx.Build(pctx, android.BuildParams{
385 Description: "Repackaging zip file " + destDir + " for snapshot " + ctx.ModuleName(),
386 Rule: repackageZip,
387 Input: zipPath,
388 Output: tmpZipPath,
389 Args: map[string]string{
390 "destdir": destDir,
391 },
392 })
Paul Duffin91547182019-11-12 19:39:36 +0000393
394 // Add the repackaged zip file to the files to merge.
395 s.zipsToMerge = append(s.zipsToMerge, tmpZipPath)
396}
397
Paul Duffin9d8d6092019-12-05 18:19:29 +0000398func (s *snapshotBuilder) AddPrebuiltModule(member android.SdkMember, moduleType string) android.BpModule {
399 name := member.Name()
Paul Duffinb645ec82019-11-27 17:43:54 +0000400 if s.prebuiltModules[name] != nil {
401 panic(fmt.Sprintf("Duplicate module detected, module %s has already been added", name))
402 }
403
404 m := s.bpFile.newModule(moduleType)
405 m.AddProperty("name", name)
Paul Duffin593b3c92019-12-05 14:31:48 +0000406
407 // Extract visibility information from a member variant. All variants have the same
408 // visibility so it doesn't matter which one is used.
409 visibility := android.EffectiveVisibilityRules(s.ctx, member.Variants()[0])
410 if len(visibility) != 0 {
411 m.AddProperty("visibility", visibility)
412 }
413
Paul Duffine44358f2019-11-26 18:04:12 +0000414 addHostDeviceSupportedProperties(&s.sdk.ModuleBase, m)
Paul Duffinb645ec82019-11-27 17:43:54 +0000415
416 s.prebuiltModules[name] = m
417 s.prebuiltOrder = append(s.prebuiltOrder, m)
418 return m
Paul Duffin0e0cf1d2019-11-12 19:39:25 +0000419}
420
Paul Duffine44358f2019-11-26 18:04:12 +0000421func addHostDeviceSupportedProperties(module *android.ModuleBase, bpModule *bpModule) {
422 if !module.DeviceSupported() {
423 bpModule.AddProperty("device_supported", false)
424 }
425 if module.HostSupported() {
426 bpModule.AddProperty("host_supported", true)
427 }
428}
429
Paul Duffinb645ec82019-11-27 17:43:54 +0000430// Get a versioned name appropriate for the SDK snapshot version being taken.
431func (s *snapshotBuilder) versionedSdkMemberName(unversionedName string) string {
Paul Duffin0e0cf1d2019-11-12 19:39:25 +0000432 return versionedSdkMemberName(s.ctx, unversionedName, s.version)
433}
Paul Duffinb645ec82019-11-27 17:43:54 +0000434
435func (s *snapshotBuilder) versionedSdkMemberNames(members []string) []string {
436 var references []string = nil
437 for _, m := range members {
438 references = append(references, s.versionedSdkMemberName(m))
439 }
440 return references
441}
Paul Duffin13879572019-11-28 14:31:38 +0000442
443var _ android.SdkMember = (*sdkMember)(nil)
444
445type sdkMember struct {
446 memberType android.SdkMemberType
447 name string
448 variants []android.SdkAware
449}
450
451func (m *sdkMember) Name() string {
452 return m.name
453}
454
455func (m *sdkMember) Variants() []android.SdkAware {
456 return m.variants
457}