1 /*
2  * Copyright (C) 2015 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #include "compile/IdAssigner.h"
18 
19 #include "test/Test.h"
20 
21 namespace aapt {
22 
23 struct IdAssignerTests : public ::testing::Test {
SetUpaapt::IdAssignerTests24   void SetUp() override {
25     context = test::ContextBuilder().SetCompilationPackage("android").SetPackageId(0x01).Build();
26   }
27   std::unique_ptr<IAaptContext> context;
28 };
29 
30 ::testing::AssertionResult VerifyIds(ResourceTable* table);
31 
TEST_F(IdAssignerTests,AssignIds)32 TEST_F(IdAssignerTests, AssignIds) {
33   auto table = test::ResourceTableBuilder()
34                    .AddSimple("android:attr/foo")
35                    .AddSimple("android:attr/bar")
36                    .AddSimple("android:id/foo")
37                    .Build();
38   IdAssigner assigner;
39 
40   ASSERT_TRUE(assigner.Consume(context.get(), table.get()));
41   ASSERT_TRUE(VerifyIds(table.get()));
42 }
43 
TEST_F(IdAssignerTests,AssignIdsWithReservedIds)44 TEST_F(IdAssignerTests, AssignIdsWithReservedIds) {
45   auto table = test::ResourceTableBuilder()
46                    .AddSimple("android:id/foo", ResourceId(0x01010000))
47                    .AddSimple("android:dimen/two")
48                    .AddSimple("android:integer/three")
49                    .AddSimple("android:string/five")
50                    .AddSimple("android:attr/fun", ResourceId(0x01040000))
51                    .AddSimple("android:attr/foo", ResourceId(0x01040006))
52                    .AddSimple("android:attr/bar")
53                    .AddSimple("android:attr/baz")
54                    .Build();
55 
56   IdAssigner assigner;
57   ASSERT_TRUE(assigner.Consume(context.get(), table.get()));
58   ASSERT_TRUE(VerifyIds(table.get()));
59 
60   std::optional<ResourceTable::SearchResult> maybe_result;
61 
62   // Expect to fill in the gaps between 0x0101XXXX and 0x0104XXXX.
63 
64   maybe_result = table->FindResource(test::ParseNameOrDie("android:dimen/two"));
65   ASSERT_TRUE(maybe_result);
66   EXPECT_EQ(0x01020000, maybe_result.value().entry->id);
67 
68   maybe_result =
69       table->FindResource(test::ParseNameOrDie("android:integer/three"));
70   ASSERT_TRUE(maybe_result);
71   EXPECT_EQ(0x01030000, maybe_result.value().entry->id);
72 
73   // Expect to bypass the reserved 0x0104XXXX IDs and use the next 0x0105XXXX
74   // IDs.
75 
76   maybe_result =
77       table->FindResource(test::ParseNameOrDie("android:string/five"));
78   ASSERT_TRUE(maybe_result);
79   EXPECT_EQ(0x01050000, maybe_result.value().entry->id);
80 
81   // Expect to fill in the gaps between 0x01040000 and 0x01040006.
82 
83   maybe_result = table->FindResource(test::ParseNameOrDie("android:attr/bar"));
84   ASSERT_TRUE(maybe_result);
85   EXPECT_EQ(0x01040001, maybe_result.value().entry->id);
86 
87   maybe_result = table->FindResource(test::ParseNameOrDie("android:attr/baz"));
88   ASSERT_TRUE(maybe_result);
89   EXPECT_EQ(0x01040002, maybe_result.value().entry->id);
90 }
91 
TEST_F(IdAssignerTests,FailWhenNonUniqueIdsAssigned)92 TEST_F(IdAssignerTests, FailWhenNonUniqueIdsAssigned) {
93   auto table = test::ResourceTableBuilder()
94                    .AddSimple("android:attr/foo", ResourceId(0x01040006))
95                    .AddSimple("android:attr/bar", ResourceId(0x01040006))
96                    .Build();
97   IdAssigner assigner;
98   ASSERT_FALSE(assigner.Consume(context.get(), table.get()));
99 }
100 
TEST_F(IdAssignerTests,FailWhenNonUniqueTypeIdsAssigned)101 TEST_F(IdAssignerTests, FailWhenNonUniqueTypeIdsAssigned) {
102   auto table = test::ResourceTableBuilder()
103                    .AddSimple("android:string/foo", ResourceId(0x01040000))
104                    .AddSimple("android:attr/bar", ResourceId(0x01040006))
105                    .Build();
106   IdAssigner assigner;
107   ASSERT_FALSE(assigner.Consume(context.get(), table.get()));
108 }
109 
TEST_F(IdAssignerTests,FailWhenTypeHasTwoNonStagedIds)110 TEST_F(IdAssignerTests, FailWhenTypeHasTwoNonStagedIds) {
111   auto table = test::ResourceTableBuilder()
112                    .AddSimple("android:attr/foo", ResourceId(0x01050000))
113                    .AddSimple("android:attr/bar", ResourceId(0x01040006))
114                    .Build();
115   IdAssigner assigner;
116   ASSERT_FALSE(assigner.Consume(context.get(), table.get()));
117 }
118 
TEST_F(IdAssignerTests,FailWhenTypeHasTwoNonStagedIdsRegardlessOfStagedId)119 TEST_F(IdAssignerTests, FailWhenTypeHasTwoNonStagedIdsRegardlessOfStagedId) {
120   auto table = test::ResourceTableBuilder()
121                    .AddSimple("android:attr/foo", ResourceId(0x01050000))
122                    .AddSimple("android:attr/bar", ResourceId(0x01ff0006))
123                    .Add(NewResourceBuilder("android:attr/staged_baz")
124                             .SetId(0x01ff0000)
125                             .SetVisibility({.staged_api = true})
126                             .Build())
127                    .Build();
128   IdAssigner assigner;
129   ASSERT_FALSE(assigner.Consume(context.get(), table.get()));
130 }
131 
TEST_F(IdAssignerTests,AssignIdsWithIdMap)132 TEST_F(IdAssignerTests, AssignIdsWithIdMap) {
133   auto table = test::ResourceTableBuilder()
134                    .AddSimple("android:attr/foo")
135                    .AddSimple("android:attr/bar")
136                    .Build();
137   std::unordered_map<ResourceName, ResourceId> id_map = {
138       {test::ParseNameOrDie("android:attr/foo"), ResourceId(0x01010002)}};
139   IdAssigner assigner(&id_map);
140   ASSERT_TRUE(assigner.Consume(context.get(), table.get()));
141   ASSERT_TRUE(VerifyIds(table.get()));
142   auto result = table->FindResource(test::ParseNameOrDie("android:attr/foo"));
143   ASSERT_TRUE(result);
144 
145   const ResourceTable::SearchResult& search_result = result.value();
146   EXPECT_EQ(0x01010002, search_result.entry->id);
147 }
148 
TEST_F(IdAssignerTests,UseAllEntryIds)149 TEST_F(IdAssignerTests, UseAllEntryIds) {
150   ResourceTable table;
151   const size_t max_entry_id = std::numeric_limits<uint16_t>::max();
152   for (size_t i = 0; i <= max_entry_id; i++) {
153     ASSERT_TRUE(
154         table.AddResource(NewResourceBuilder("android:attr/res" + std::to_string(i)).Build(),
155                           context->GetDiagnostics()));
156   }
157   IdAssigner assigner;
158   ASSERT_TRUE(assigner.Consume(context.get(), &table));
159 }
160 
TEST_F(IdAssignerTests,ExaustEntryIds)161 TEST_F(IdAssignerTests, ExaustEntryIds) {
162   ResourceTable table;
163   const size_t max_entry_id = std::numeric_limits<uint16_t>::max() + 1u;
164   for (size_t i = 0; i <= max_entry_id; i++) {
165     ASSERT_TRUE(
166         table.AddResource(NewResourceBuilder("android:attr/res" + std::to_string(i)).Build(),
167                           context->GetDiagnostics()));
168   }
169   IdAssigner assigner;
170   ASSERT_FALSE(assigner.Consume(context.get(), &table));
171 }
172 
TEST_F(IdAssignerTests,ExaustEntryIdsLastIdIsPublic)173 TEST_F(IdAssignerTests, ExaustEntryIdsLastIdIsPublic) {
174   ResourceTable table;
175   ASSERT_TRUE(table.AddResource(NewResourceBuilder("android:attr/res").SetId(0x0101ffff).Build(),
176                                 context->GetDiagnostics()));
177   const size_t max_entry_id = std::numeric_limits<uint16_t>::max();
178   for (size_t i = 0; i <= max_entry_id; i++) {
179     ASSERT_TRUE(
180         table.AddResource(NewResourceBuilder("android:attr/res" + std::to_string(i)).Build(),
181                           context->GetDiagnostics()));
182   }
183   IdAssigner assigner;
184   ASSERT_FALSE(assigner.Consume(context.get(), &table));
185 }
186 
VerifyIds(ResourceTable * table)187 ::testing::AssertionResult VerifyIds(ResourceTable* table) {
188   std::set<ResourceId> seen_ids;
189   for (auto& package : table->packages) {
190     for (auto& type : package->types) {
191       for (auto& entry : type->entries) {
192         if (!entry->id) {
193           return ::testing::AssertionFailure()
194                  << "resource " << ResourceNameRef(package->name, type->named_type, entry->name)
195                  << " has no ID";
196         }
197         if (!seen_ids.insert(entry->id.value()).second) {
198           return ::testing::AssertionFailure()
199                  << "resource " << ResourceNameRef(package->name, type->named_type, entry->name)
200                  << " has a non-unique ID" << std::hex << entry->id.value() << std::dec;
201         }
202       }
203     }
204   }
205 
206   return ::testing::AssertionSuccess() << "all IDs are unique and assigned";
207 }
208 
209 }  // namespace aapt
210