OLD | NEW |
1 // Copyright 2014 The Chromium Authors. All rights reserved. | 1 // Copyright 2014 The Chromium Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "ui/display/util/edid_parser.h" | 5 #include "ui/display/util/edid_parser.h" |
6 | 6 |
7 #include "base/memory/scoped_ptr.h" | 7 #include "base/memory/scoped_ptr.h" |
8 #include "testing/gtest/include/gtest/gtest.h" | 8 #include "testing/gtest/include/gtest/gtest.h" |
| 9 #include "ui/gfx/geometry/size.h" |
9 | 10 |
10 namespace ui { | 11 namespace ui { |
11 | 12 |
12 namespace { | 13 namespace { |
13 | 14 |
14 // Returns the number of characters in the string literal but doesn't count its | 15 // Returns the number of characters in the string literal but doesn't count its |
15 // terminator NULL byte. | 16 // terminator NULL byte. |
16 #define charsize(str) (arraysize(str) - 1) | 17 #define charsize(str) (arraysize(str) - 1) |
17 | 18 |
18 // Sample EDID data extracted from real devices. | 19 // Sample EDID data extracted from real devices. |
(...skipping 67 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
86 const unsigned char kLP2565B[] = | 87 const unsigned char kLP2565B[] = |
87 "\x00\xFF\xFF\xFF\xFF\xFF\xFF\x00\x22\xF0\x75\x26\x01\x01\x01\x01" | 88 "\x00\xFF\xFF\xFF\xFF\xFF\xFF\x00\x22\xF0\x75\x26\x01\x01\x01\x01" |
88 "\x02\x12\x01\x03\x6E\x34\x21\x78\xEE\xEF\x95\xA3\x54\x4C\x9B\x26" | 89 "\x02\x12\x01\x03\x6E\x34\x21\x78\xEE\xEF\x95\xA3\x54\x4C\x9B\x26" |
89 "\x0F\x50\x54\xA5\x6B\x80\x81\x40\x71\x00\xA9\x00\xA9\x40\xA9\x4F" | 90 "\x0F\x50\x54\xA5\x6B\x80\x81\x40\x71\x00\xA9\x00\xA9\x40\xA9\x4F" |
90 "\xB3\x00\xD1\xC0\xD1\x00\x28\x3C\x80\xA0\x70\xB0\x23\x40\x30\x20" | 91 "\xB3\x00\xD1\xC0\xD1\x00\x28\x3C\x80\xA0\x70\xB0\x23\x40\x30\x20" |
91 "\x36\x00\x07\x44\x21\x00\x00\x1A\x00\x00\x00\xFD\x00\x30\x55\x1E" | 92 "\x36\x00\x07\x44\x21\x00\x00\x1A\x00\x00\x00\xFD\x00\x30\x55\x1E" |
92 "\x5E\x15\x00\x0A\x20\x20\x20\x20\x20\x20\x00\x00\x00\xFC\x00\x48" | 93 "\x5E\x15\x00\x0A\x20\x20\x20\x20\x20\x20\x00\x00\x00\xFC\x00\x48" |
93 "\x50\x20\x4C\x50\x32\x34\x36\x35\x0A\x20\x20\x20\x00\x00\x00\xFF" | 94 "\x50\x20\x4C\x50\x32\x34\x36\x35\x0A\x20\x20\x20\x00\x00\x00\xFF" |
94 "\x00\x43\x4E\x4B\x38\x30\x32\x30\x34\x48\x4D\x0A\x20\x20\x00\x45"; | 95 "\x00\x43\x4E\x4B\x38\x30\x32\x30\x34\x48\x4D\x0A\x20\x20\x00\x45"; |
95 | 96 |
| 97 void Reset(gfx::Size* pixel, gfx::Size* size) { |
| 98 pixel->SetSize(0, 0); |
| 99 size->SetSize(0, 0); |
| 100 } |
| 101 |
96 } // namespace | 102 } // namespace |
97 | 103 |
98 TEST(EDIDParserTest, ParseOverscanFlag) { | 104 TEST(EDIDParserTest, ParseOverscanFlag) { |
99 bool flag = false; | 105 bool flag = false; |
100 std::vector<uint8_t> edid( | 106 std::vector<uint8_t> edid( |
101 kNormalDisplay, kNormalDisplay + charsize(kNormalDisplay)); | 107 kNormalDisplay, kNormalDisplay + charsize(kNormalDisplay)); |
102 EXPECT_FALSE(ParseOutputOverscanFlag(edid, &flag)); | 108 EXPECT_FALSE(ParseOutputOverscanFlag(edid, &flag)); |
103 | 109 |
104 flag = false; | 110 flag = false; |
105 edid.assign(kInternalDisplay, kInternalDisplay + charsize(kInternalDisplay)); | 111 edid.assign(kInternalDisplay, kInternalDisplay + charsize(kInternalDisplay)); |
(...skipping 33 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
139 data[126] = '\x01'; | 145 data[126] = '\x01'; |
140 EXPECT_FALSE(ParseOutputOverscanFlag(data, &flag)); | 146 EXPECT_FALSE(ParseOutputOverscanFlag(data, &flag)); |
141 | 147 |
142 data.assign(150, '\0'); | 148 data.assign(150, '\0'); |
143 data[126] = '\x01'; | 149 data[126] = '\x01'; |
144 EXPECT_FALSE(ParseOutputOverscanFlag(data, &flag)); | 150 EXPECT_FALSE(ParseOutputOverscanFlag(data, &flag)); |
145 } | 151 } |
146 | 152 |
147 TEST(EDIDParserTest, ParseEDID) { | 153 TEST(EDIDParserTest, ParseEDID) { |
148 uint16_t manufacturer_id = 0; | 154 uint16_t manufacturer_id = 0; |
| 155 uint16_t product_code = 0; |
149 std::string human_readable_name; | 156 std::string human_readable_name; |
150 std::vector<uint8_t> edid( | 157 std::vector<uint8_t> edid( |
151 kNormalDisplay, kNormalDisplay + charsize(kNormalDisplay)); | 158 kNormalDisplay, kNormalDisplay + charsize(kNormalDisplay)); |
152 EXPECT_TRUE(ParseOutputDeviceData( | 159 gfx::Size pixel; |
153 edid, &manufacturer_id, &human_readable_name)); | 160 gfx::Size size; |
| 161 EXPECT_TRUE(ParseOutputDeviceData(edid, &manufacturer_id, &product_code, |
| 162 &human_readable_name, &pixel, &size)); |
154 EXPECT_EQ(0x22f0u, manufacturer_id); | 163 EXPECT_EQ(0x22f0u, manufacturer_id); |
| 164 EXPECT_EQ(0x286cu, product_code); |
155 EXPECT_EQ("HP ZR30w", human_readable_name); | 165 EXPECT_EQ("HP ZR30w", human_readable_name); |
| 166 EXPECT_EQ("2560x1600", pixel.ToString()); |
| 167 EXPECT_EQ("641x400", size.ToString()); |
156 | 168 |
157 manufacturer_id = 0; | 169 manufacturer_id = 0; |
| 170 product_code = 0; |
158 human_readable_name.clear(); | 171 human_readable_name.clear(); |
| 172 Reset(&pixel, &size); |
159 edid.assign(kInternalDisplay, kInternalDisplay + charsize(kInternalDisplay)); | 173 edid.assign(kInternalDisplay, kInternalDisplay + charsize(kInternalDisplay)); |
160 EXPECT_TRUE(ParseOutputDeviceData(edid, &manufacturer_id, NULL)); | 174 |
| 175 EXPECT_TRUE(ParseOutputDeviceData(edid, &manufacturer_id, &product_code, |
| 176 nullptr, &pixel, &size)); |
161 EXPECT_EQ(0x4ca3u, manufacturer_id); | 177 EXPECT_EQ(0x4ca3u, manufacturer_id); |
| 178 EXPECT_EQ(0x3142u, product_code); |
162 EXPECT_EQ("", human_readable_name); | 179 EXPECT_EQ("", human_readable_name); |
| 180 EXPECT_EQ("1280x800", pixel.ToString()); |
| 181 EXPECT_EQ("261x163", size.ToString()); |
163 | 182 |
164 // Internal display doesn't have name. | 183 // Internal display doesn't have name. |
165 EXPECT_TRUE(ParseOutputDeviceData(edid, NULL, &human_readable_name)); | 184 EXPECT_TRUE(ParseOutputDeviceData(edid, nullptr, nullptr, |
| 185 &human_readable_name, &pixel, &size)); |
166 EXPECT_TRUE(human_readable_name.empty()); | 186 EXPECT_TRUE(human_readable_name.empty()); |
167 | 187 |
168 manufacturer_id = 0; | 188 manufacturer_id = 0; |
| 189 product_code = 0; |
169 human_readable_name.clear(); | 190 human_readable_name.clear(); |
| 191 Reset(&pixel, &size); |
170 edid.assign(kOverscanDisplay, kOverscanDisplay + charsize(kOverscanDisplay)); | 192 edid.assign(kOverscanDisplay, kOverscanDisplay + charsize(kOverscanDisplay)); |
171 EXPECT_TRUE(ParseOutputDeviceData( | 193 EXPECT_TRUE(ParseOutputDeviceData(edid, &manufacturer_id, &product_code, |
172 edid, &manufacturer_id, &human_readable_name)); | 194 &human_readable_name, &pixel, &size)); |
173 EXPECT_EQ(0x4c2du, manufacturer_id); | 195 EXPECT_EQ(0x4c2du, manufacturer_id); |
| 196 EXPECT_EQ(0x08feu, product_code); |
174 EXPECT_EQ("SAMSUNG", human_readable_name); | 197 EXPECT_EQ("SAMSUNG", human_readable_name); |
| 198 EXPECT_EQ("1920x1080", pixel.ToString()); |
| 199 EXPECT_EQ("160x90", size.ToString()); |
175 } | 200 } |
176 | 201 |
177 TEST(EDIDParserTest, ParseBrokenEDID) { | 202 TEST(EDIDParserTest, ParseBrokenEDID) { |
178 uint16_t manufacturer_id = 0; | 203 uint16_t manufacturer_id = 0; |
| 204 uint16_t product_code = 0; |
179 std::string human_readable_name; | 205 std::string human_readable_name; |
180 std::vector<uint8_t> edid; | 206 std::vector<uint8_t> edid; |
181 | 207 |
| 208 gfx::Size dummy; |
| 209 |
182 // length == 0 | 210 // length == 0 |
183 EXPECT_FALSE(ParseOutputDeviceData( | 211 EXPECT_FALSE(ParseOutputDeviceData(edid, &manufacturer_id, &product_code, |
184 edid, &manufacturer_id, &human_readable_name)); | 212 &human_readable_name, &dummy, &dummy)); |
185 | 213 |
186 // name is broken. Copying kNormalDisplay and substitute its name data by | 214 // name is broken. Copying kNormalDisplay and substitute its name data by |
187 // some control code. | 215 // some control code. |
188 edid.assign(kNormalDisplay, kNormalDisplay + charsize(kNormalDisplay)); | 216 edid.assign(kNormalDisplay, kNormalDisplay + charsize(kNormalDisplay)); |
189 | 217 |
190 // display's name data is embedded in byte 95-107 in this specific example. | 218 // display's name data is embedded in byte 95-107 in this specific example. |
191 // Fix here too when the contents of kNormalDisplay is altered. | 219 // Fix here too when the contents of kNormalDisplay is altered. |
192 edid[97] = '\x1b'; | 220 edid[97] = '\x1b'; |
193 EXPECT_FALSE(ParseOutputDeviceData( | 221 EXPECT_FALSE(ParseOutputDeviceData(edid, &manufacturer_id, nullptr, |
194 edid, &manufacturer_id, &human_readable_name)); | 222 &human_readable_name, &dummy, &dummy)); |
195 | 223 |
196 // If |human_readable_name| isn't specified, it skips parsing the name. | 224 // If |human_readable_name| isn't specified, it skips parsing the name. |
197 manufacturer_id = 0; | 225 manufacturer_id = 0; |
198 EXPECT_TRUE(ParseOutputDeviceData(edid, &manufacturer_id, NULL)); | 226 product_code = 0; |
| 227 EXPECT_TRUE(ParseOutputDeviceData(edid, &manufacturer_id, &product_code, |
| 228 nullptr, &dummy, &dummy)); |
199 EXPECT_EQ(0x22f0u, manufacturer_id); | 229 EXPECT_EQ(0x22f0u, manufacturer_id); |
| 230 EXPECT_EQ(0x286cu, product_code); |
200 } | 231 } |
201 | 232 |
202 TEST(EDIDParserTest, GetDisplayId) { | 233 TEST(EDIDParserTest, GetDisplayId) { |
203 // EDID of kLP2565A and B are slightly different but actually the same device. | 234 // EDID of kLP2565A and B are slightly different but actually the same device. |
204 int64_t id1 = -1; | 235 int64_t id1 = -1; |
205 int64_t id2 = -1; | 236 int64_t id2 = -1; |
| 237 int64_t product_id1 = -1; |
| 238 int64_t product_id2 = -1; |
206 std::vector<uint8_t> edid(kLP2565A, kLP2565A + charsize(kLP2565A)); | 239 std::vector<uint8_t> edid(kLP2565A, kLP2565A + charsize(kLP2565A)); |
207 EXPECT_TRUE(GetDisplayIdFromEDID(edid, 0, &id1)); | 240 EXPECT_TRUE(GetDisplayIdFromEDID(edid, 0, &id1, &product_id1)); |
208 edid.assign(kLP2565B, kLP2565B + charsize(kLP2565B)); | 241 edid.assign(kLP2565B, kLP2565B + charsize(kLP2565B)); |
209 EXPECT_TRUE(GetDisplayIdFromEDID(edid, 0, &id2)); | 242 EXPECT_TRUE(GetDisplayIdFromEDID(edid, 0, &id2, &product_id2)); |
210 EXPECT_EQ(id1, id2); | 243 EXPECT_EQ(id1, id2); |
| 244 // The product code in the two EDIDs varies. |
| 245 EXPECT_NE(product_id1, product_id2); |
| 246 EXPECT_EQ(0x22f02676, product_id1); |
| 247 EXPECT_EQ(0x22f02675, product_id2); |
211 EXPECT_NE(-1, id1); | 248 EXPECT_NE(-1, id1); |
| 249 EXPECT_NE(-1, product_id1); |
212 } | 250 } |
213 | 251 |
214 TEST(EDIDParserTest, GetDisplayIdFromInternal) { | 252 TEST(EDIDParserTest, GetDisplayIdFromInternal) { |
215 int64_t id = -1; | 253 int64_t id = -1; |
| 254 int64_t product_id = -1; |
216 std::vector<uint8_t> edid( | 255 std::vector<uint8_t> edid( |
217 kInternalDisplay, kInternalDisplay + charsize(kInternalDisplay)); | 256 kInternalDisplay, kInternalDisplay + charsize(kInternalDisplay)); |
218 EXPECT_TRUE(GetDisplayIdFromEDID(edid, 0, &id)); | 257 EXPECT_TRUE(GetDisplayIdFromEDID(edid, 0, &id, &product_id)); |
219 EXPECT_NE(-1, id); | 258 EXPECT_NE(-1, id); |
| 259 EXPECT_NE(-1, product_id); |
220 } | 260 } |
221 | 261 |
222 TEST(EDIDParserTest, GetDisplayIdFailure) { | 262 TEST(EDIDParserTest, GetDisplayIdFailure) { |
223 int64_t id = -1; | 263 int64_t id = -1; |
| 264 int64_t product_id = -1; |
224 std::vector<uint8_t> edid; | 265 std::vector<uint8_t> edid; |
225 EXPECT_FALSE(GetDisplayIdFromEDID(edid, 0, &id)); | 266 EXPECT_FALSE(GetDisplayIdFromEDID(edid, 0, &id, &product_id)); |
226 EXPECT_EQ(-1, id); | 267 EXPECT_EQ(-1, id); |
| 268 EXPECT_EQ(-1, product_id); |
227 } | 269 } |
228 | 270 |
229 } // namespace ui | 271 } // namespace ui |
OLD | NEW |