1 /*
2 * Copyright (c) 2017-2020, The Linux Foundation. All rights reserved.
3 *
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions are
6 * met:
7 *     * Redistributions of source code must retain the above copyright
8 *       notice, this list of conditions and the following disclaimer.
9 *     * Redistributions in binary form must reproduce the above
10 *       copyright notice, this list of conditions and the following
11 *       disclaimer in the documentation and/or other materials provided
12 *       with the distribution.
13 *     * Neither the name of The Linux Foundation nor the names of its
14 *       contributors may be used to endorse or promote products derived
15 *       from this software without specific prior written permission.
16 *
17 * THIS SOFTWARE IS PROVIDED "AS IS" AND ANY EXPRESS OR IMPLIED
18 * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
19 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT
20 * ARE DISCLAIMED.  IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS
21 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
22 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
23 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
24 * BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
25 * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE
26 * OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN
27 * IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28 */
29 
30 #include <cutils/properties.h>
31 #include <sys/mman.h>
32 #include <sys/types.h>
33 #include <sys/stat.h>
34 #include <utils/constants.h>
35 #include <utils/debug.h>
36 #include <utils/formats.h>
37 #include <algorithm>
38 #include <array>
39 #include <sstream>
40 #include <string>
41 #include <fstream>
42 
43 #include "hwc_display_pluggable_test.h"
44 #include "hwc_debugger.h"
45 
46 #define __CLASS__ "HWCDisplayPluggableTest"
47 
48 namespace sdm {
49 
50 using std::array;
51 
Create(CoreInterface * core_intf,HWCBufferAllocator * buffer_allocator,HWCCallbacks * callbacks,HWCDisplayEventHandler * event_handler,qService::QService * qservice,hwc2_display_t id,int32_t sdm_id,uint32_t panel_bpp,uint32_t pattern_type,HWCDisplay ** hwc_display)52 int HWCDisplayPluggableTest::Create(CoreInterface *core_intf, HWCBufferAllocator *buffer_allocator,
53                                     HWCCallbacks *callbacks, HWCDisplayEventHandler *event_handler,
54                                     qService::QService *qservice, hwc2_display_t id,
55                                     int32_t sdm_id, uint32_t panel_bpp, uint32_t pattern_type,
56                                     HWCDisplay **hwc_display) {
57   HWCDisplay *hwc_pluggable_test = new HWCDisplayPluggableTest(core_intf, buffer_allocator,
58                                                                callbacks, event_handler, qservice,
59                                                                id, sdm_id, panel_bpp, pattern_type);
60 
61   int status = hwc_pluggable_test->Init();
62   if (status) {
63     delete hwc_pluggable_test;
64     return status;
65   }
66 
67   *hwc_display = hwc_pluggable_test;
68 
69   DLOGD("Pluggable panel_bpp %d, pattern_type %d", panel_bpp, pattern_type);
70 
71   return status;
72 }
73 
Destroy(HWCDisplay * hwc_display)74 void HWCDisplayPluggableTest::Destroy(HWCDisplay *hwc_display) {
75   // Flush the display to have outstanding fences signaled.
76   hwc_display->Flush();
77   hwc_display->Deinit();
78 
79   delete hwc_display;
80 }
81 
HWCDisplayPluggableTest(CoreInterface * core_intf,HWCBufferAllocator * buffer_allocator,HWCCallbacks * callbacks,HWCDisplayEventHandler * event_handler,qService::QService * qservice,hwc2_display_t id,int32_t sdm_id,uint32_t panel_bpp,uint32_t pattern_type)82 HWCDisplayPluggableTest::HWCDisplayPluggableTest(CoreInterface *core_intf,
83                                                  HWCBufferAllocator *buffer_allocator,
84                                                  HWCCallbacks *callbacks,
85                                                  HWCDisplayEventHandler *event_handler,
86                                                  qService::QService *qservice, hwc2_display_t id,
87                                                  int32_t sdm_id, uint32_t panel_bpp,
88                                                  uint32_t pattern_type)
89   : HWCDisplay(core_intf, buffer_allocator, callbacks, event_handler, qservice, kPluggable, id,
90                sdm_id, DISPLAY_CLASS_PLUGGABLE),
91     panel_bpp_(panel_bpp), pattern_type_(pattern_type) {
92 }
93 
Init()94 int HWCDisplayPluggableTest::Init() {
95   uint32_t pluggable_width = 0;
96   uint32_t pluggable_height = 0;
97 
98   int status = HWCDisplay::Init();
99   if (status) {
100       DLOGE("HWCDisplayPluggableTest::Init  status = %d ", status);
101     return status;
102   }
103 
104   status = CreateLayerStack();
105   if (status) {
106     Deinit();
107     return status;
108   }
109 
110   DisplayError error = HWCDisplay::GetMixerResolution(&pluggable_width, &pluggable_height);
111   if (error != kErrorNone) {
112     Deinit();
113     return -EINVAL;
114   }
115 
116   status = HWCDisplay::SetFrameBufferResolution(pluggable_width, pluggable_height);
117   if (status) {
118     Deinit();
119     DLOGE("HWCDisplayPluggableTest:: set fb resolution status = %d ", status);
120     return status;
121   }
122 
123   return status;
124 }
125 
Deinit()126 int HWCDisplayPluggableTest::Deinit() {
127   DestroyLayerStack();
128   return HWCDisplay::Deinit();
129 }
130 
131 
Validate(uint32_t * out_num_types,uint32_t * out_num_requests)132 HWC2::Error HWCDisplayPluggableTest::Validate(uint32_t *out_num_types, uint32_t *out_num_requests) {
133   auto status = HWC2::Error::None;
134   if (active_secure_sessions_[kSecureDisplay] || display_paused_) {
135     MarkLayersForGPUBypass();
136     return status;
137   }
138 
139   if (layer_set_.empty()) {
140       flush_ = true;
141       return status;
142   }
143 
144   if (shutdown_pending_) {
145     return status;
146   }
147   DisplayError error = display_intf_->Prepare(&layer_stack_);
148   if (error != kErrorNone) {
149     if (error == kErrorShutDown) {
150       shutdown_pending_ = true;
151     } else if (error != kErrorPermission) {
152       DLOGE("Prepare failed. Error = %d", error);
153       // To prevent surfaceflinger infinite wait, flush the previous frame during Commit()
154       // so that previous buffer and fences are released, and override the error.
155       flush_ = true;
156     }
157   }
158 
159   MarkLayersForGPUBypass();
160 
161   return  status;
162 }
163 
Present(shared_ptr<Fence> * out_retire_fence)164 HWC2::Error HWCDisplayPluggableTest::Present(shared_ptr<Fence> *out_retire_fence) {
165   auto status = HWC2::Error::None;
166 
167   if (active_secure_sessions_[kSecureDisplay]) {
168     return status;
169   }
170 
171   if (display_paused_) {
172     DisplayError error = display_intf_->Flush(&layer_stack_);
173     validated_ = false;
174     if (error != kErrorNone) {
175       DLOGE("Flush failed. Error = %d", error);
176     }
177   }
178 
179   if (shutdown_pending_) {
180     return status;
181   }
182 
183   DumpInputBuffer();
184 
185   if (!flush_) {
186     DisplayError error = kErrorUndefined;
187     error = display_intf_->Commit(&layer_stack_);
188     if (error == kErrorNone) {
189       // A commit is successfully submitted, start flushing on failure now onwards.
190       flush_on_error_ = true;
191     } else if (error == kErrorShutDown) {
192       shutdown_pending_ = true;
193       status = HWC2::Error::Unsupported;
194     } else if (error == kErrorNotValidated) {
195       status = HWC2::Error::NotValidated;
196     } else if (error != kErrorPermission) {
197       DLOGE("Commit failed. Error = %d", error);
198         // To prevent surfaceflinger infinite wait, flush the previous frame during Commit()
199         // so that previous buffer and fences are released, and override the error.
200         flush_ = true;
201     }
202   }
203   PostCommit(out_retire_fence);
204   return status;
205 }
206 
Perform(uint32_t operation,...)207 int HWCDisplayPluggableTest::Perform(uint32_t operation, ...) {
208   return 0;
209 }
210 
DumpInputBuffer()211 void HWCDisplayPluggableTest::DumpInputBuffer() {
212   if (!dump_frame_count_ || flush_ || !dump_input_layers_) {
213     return;
214   }
215 
216   const char *dir_path = "/data/vendor/display/frame_dump_external";
217   uint32_t width = buffer_info_.alloc_buffer_info.aligned_width;
218   uint32_t height = buffer_info_.alloc_buffer_info.aligned_height;
219   string format_str = GetFormatString(buffer_info_.buffer_config.format);
220 
221   char *buffer = reinterpret_cast<char *>(mmap(NULL, buffer_info_.alloc_buffer_info.size,
222                                                 PROT_READ|PROT_WRITE, MAP_SHARED,
223                                                 buffer_info_.alloc_buffer_info.fd, 0));
224   if (buffer == MAP_FAILED) {
225     DLOGW("mmap failed. err = %d", errno);
226     return;
227   }
228 
229   if (mkdir(dir_path, 0777) != 0 && errno != EEXIST) {
230     DLOGW("Failed to create %s directory errno = %d, desc = %s", dir_path, errno, strerror(errno));
231     return;
232   }
233 
234   // if directory exists already, need to explicitly change the permission.
235   if (errno == EEXIST && chmod(dir_path, 0777) != 0) {
236     DLOGW("Failed to change permissions on %s directory", dir_path);
237     return;
238   }
239 
240   if (buffer) {
241     std::stringstream dump_file_name;
242     dump_file_name << dir_path;
243     dump_file_name << "/input_layer_" << width << "x" << height << "_" << format_str << ".raw";
244 
245     std::fstream fs;
246     fs.open(dump_file_name.str().c_str(), std::fstream::in | std::fstream::out | std::fstream::app);
247     if (!fs.is_open()) {
248       DLOGI("File open failed %s", dump_file_name.str().c_str());
249       return;
250     }
251 
252     fs.write(buffer, (std::streamsize)buffer_info_.alloc_buffer_info.size);
253     fs.close();
254 
255     DLOGI("Frame Dump %s: is successful", dump_file_name.str().c_str());
256   }
257 
258   // Dump only once as the content is going to be same for all draw cycles
259   if (dump_frame_count_) {
260     dump_frame_count_ = 0;
261   }
262 
263   if (munmap(buffer, buffer_info_.alloc_buffer_info.size) != 0) {
264     DLOGW("munmap failed. err = %d", errno);
265     return;
266   }
267 }
268 
CalcCRC(uint32_t color_val,std::bitset<16> * crc_data)269 void HWCDisplayPluggableTest::CalcCRC(uint32_t color_val, std::bitset<16> *crc_data) {
270   std::bitset<16> color = {};
271   std::bitset<16> temp_crc = {};
272 
273   switch (panel_bpp_) {
274     case kDisplayBpp18:
275       color = (color_val & 0xFC) << 8;
276       break;
277     case kDisplayBpp24:
278       color = color_val << 8;
279       break;
280     case kDisplayBpp30:
281       color = color_val << 6;
282       break;
283     default:
284       return;
285   }
286 
287   temp_crc[15] = (*crc_data)[0] ^ (*crc_data)[1] ^ (*crc_data)[2] ^ (*crc_data)[3] ^
288                  (*crc_data)[4] ^ (*crc_data)[5] ^ (*crc_data)[6] ^ (*crc_data)[7] ^
289                  (*crc_data)[8] ^ (*crc_data)[9] ^ (*crc_data)[10] ^ (*crc_data)[11] ^
290                  (*crc_data)[12] ^ (*crc_data)[14] ^ (*crc_data)[15] ^ color[0] ^ color[1] ^
291                  color[2] ^ color[3] ^ color[4] ^ color[5] ^ color[6] ^ color[7] ^ color[8] ^
292                  color[9] ^ color[10] ^ color[11] ^ color[12] ^ color[14] ^ color[15];
293 
294   temp_crc[14] = (*crc_data)[12] ^ (*crc_data)[13] ^ color[12] ^ color[13];
295   temp_crc[13] = (*crc_data)[11] ^ (*crc_data)[12] ^ color[11] ^ color[12];
296   temp_crc[12] = (*crc_data)[10] ^ (*crc_data)[11] ^ color[10] ^ color[11];
297   temp_crc[11] = (*crc_data)[9] ^ (*crc_data)[10] ^ color[9] ^ color[10];
298   temp_crc[10] = (*crc_data)[8] ^ (*crc_data)[9] ^ color[8] ^ color[9];
299   temp_crc[9] = (*crc_data)[7] ^ (*crc_data)[8] ^ color[7] ^ color[8];
300   temp_crc[8] = (*crc_data)[6] ^ (*crc_data)[7] ^ color[6] ^ color[7];
301   temp_crc[7] = (*crc_data)[5] ^ (*crc_data)[6] ^ color[5] ^ color[6];
302   temp_crc[6] = (*crc_data)[4] ^ (*crc_data)[5] ^ color[4] ^ color[5];
303   temp_crc[5] = (*crc_data)[3] ^ (*crc_data)[4] ^ color[3] ^ color[4];
304   temp_crc[4] = (*crc_data)[2] ^ (*crc_data)[3] ^ color[2] ^ color[3];
305   temp_crc[3] = (*crc_data)[1] ^ (*crc_data)[2] ^ (*crc_data)[15] ^ color[1] ^ color[2] ^ color[15];
306   temp_crc[2] = (*crc_data)[0] ^ (*crc_data)[1] ^ (*crc_data)[14] ^ color[0] ^ color[1] ^ color[14];
307 
308   temp_crc[1] = (*crc_data)[1] ^ (*crc_data)[2] ^ (*crc_data)[3] ^ (*crc_data)[4] ^ (*crc_data)[5] ^
309                 (*crc_data)[6] ^ (*crc_data)[7] ^ (*crc_data)[8] ^ (*crc_data)[9] ^
310                 (*crc_data)[10] ^ (*crc_data)[11] ^ (*crc_data)[12] ^ (*crc_data)[13] ^
311                 (*crc_data)[14] ^ color[1] ^ color[2] ^ color[3] ^ color[4] ^ color[5] ^ color[6] ^
312                 color[7] ^ color[8] ^ color[9] ^ color[10] ^ color[11] ^ color[12] ^ color[13] ^
313                 color[14];
314 
315   temp_crc[0] = (*crc_data)[0] ^ (*crc_data)[1] ^ (*crc_data)[2] ^ (*crc_data)[3] ^ (*crc_data)[4] ^
316                 (*crc_data)[5] ^ (*crc_data)[6] ^ (*crc_data)[7] ^ (*crc_data)[8] ^ (*crc_data)[9] ^
317                 (*crc_data)[10] ^ (*crc_data)[11] ^ (*crc_data)[12] ^ (*crc_data)[13] ^
318                 (*crc_data)[15] ^ color[0] ^ color[1] ^ color[2] ^ color[3] ^ color[4] ^ color[5] ^
319                 color[6] ^ color[7] ^ color[8] ^ color[9] ^ color[10] ^ color[11] ^ color[12] ^
320                 color[13] ^ color[15];
321 
322   (*crc_data) = temp_crc;
323 }
324 
FillBuffer()325 int HWCDisplayPluggableTest::FillBuffer() {
326   uint8_t *buffer = reinterpret_cast<uint8_t *>(mmap(NULL, buffer_info_.alloc_buffer_info.size,
327                                                 PROT_READ|PROT_WRITE, MAP_SHARED,
328                                                 buffer_info_.alloc_buffer_info.fd, 0));
329   if (buffer == MAP_FAILED) {
330     DLOGE("mmap failed. err = %d", errno);
331     return -EFAULT;
332   }
333 
334   switch (pattern_type_) {
335     case kPatternColorRamp:
336       GenerateColorRamp(buffer);
337       break;
338     case kPatternBWVertical:
339       GenerateBWVertical(buffer);
340       break;
341     case kPatternColorSquare:
342       GenerateColorSquare(buffer);
343       break;
344     default:
345       DLOGW("Invalid Pattern type %d", pattern_type_);
346       return -EINVAL;
347   }
348 
349   if (munmap(buffer, buffer_info_.alloc_buffer_info.size) != 0) {
350     DLOGE("munmap failed. err = %d", errno);
351     return -EFAULT;
352   }
353 
354   return 0;
355 }
356 
GetStride(LayerBufferFormat format,uint32_t width,uint32_t * stride)357 int HWCDisplayPluggableTest::GetStride(LayerBufferFormat format, uint32_t width, uint32_t *stride) {
358   switch (format) {
359   case kFormatRGBA8888:
360   case kFormatRGBA1010102:
361     *stride = width * 4;
362     break;
363   case kFormatRGB888:
364     *stride = width * 3;
365     break;
366   default:
367     DLOGW("Unsupported format type %d", format);
368     return -EINVAL;
369   }
370 
371   return 0;
372 }
373 
PixelCopy(uint32_t red,uint32_t green,uint32_t blue,uint32_t alpha,uint8_t ** buffer)374 void HWCDisplayPluggableTest::PixelCopy(uint32_t red, uint32_t green, uint32_t blue, uint32_t alpha,
375                                        uint8_t **buffer) {
376   LayerBufferFormat format = buffer_info_.buffer_config.format;
377 
378   switch (format) {
379     case kFormatRGBA8888:
380       *(*buffer)++ = UINT8(red & 0xFF);
381       *(*buffer)++ = UINT8(green & 0xFF);
382       *(*buffer)++ = UINT8(blue & 0xFF);
383       *(*buffer)++ = UINT8(alpha & 0xFF);
384       break;
385     case kFormatRGB888:
386       *(*buffer)++ = UINT8(red & 0xFF);
387       *(*buffer)++ = UINT8(green & 0xFF);
388       *(*buffer)++ = UINT8(blue & 0xFF);
389       break;
390     case kFormatRGBA1010102:
391       // Lower 8 bits of red
392       *(*buffer)++ = UINT8(red & 0xFF);
393 
394       // Upper 2 bits of Red + Lower 6 bits of green
395       *(*buffer)++ = UINT8(((green & 0x3F) << 2) | ((red >> 0x8) & 0x3));
396 
397       // Upper 4 bits of green + Lower 4 bits of blue
398       *(*buffer)++ = UINT8(((blue & 0xF) << 4) | ((green >> 6) & 0xF));
399 
400       // Upper 6 bits of blue + Lower 2 bits of alpha
401       *(*buffer)++ = UINT8(((alpha & 0x3) << 6) | ((blue >> 4) & 0x3F));
402       break;
403     default:
404       DLOGW("format not supported format = %d", format);
405       break;
406   }
407 }
408 
GenerateColorRamp(uint8_t * buffer)409 void HWCDisplayPluggableTest::GenerateColorRamp(uint8_t *buffer) {
410   uint32_t width = buffer_info_.buffer_config.width;
411   uint32_t height = buffer_info_.buffer_config.height;
412   LayerBufferFormat format = buffer_info_.buffer_config.format;
413   uint32_t aligned_width = buffer_info_.alloc_buffer_info.aligned_width;
414   uint32_t buffer_stride = 0;
415 
416   uint32_t color_ramp = 0;
417   uint32_t start_color_val = 0;
418   uint32_t step_size = 1;
419   uint32_t ramp_width = 0;
420   uint32_t ramp_height = 0;
421   uint32_t shift_by = 0;
422 
423   std::bitset<16> crc_red = {};
424   std::bitset<16> crc_green = {};
425   std::bitset<16> crc_blue = {};
426 
427   switch (panel_bpp_) {
428     case kDisplayBpp18:
429       ramp_height = 64;
430       ramp_width = 64;
431       shift_by = 2;
432       break;
433     case kDisplayBpp24:
434       ramp_height = 64;
435       ramp_width = 256;
436       break;
437     case kDisplayBpp30:
438       ramp_height = 32;
439       ramp_width = 256;
440       start_color_val = 0x180;
441       break;
442     default:
443       return;
444   }
445 
446   GetStride(format, aligned_width, &buffer_stride);
447 
448   for (uint32_t loop_height = 0; loop_height < height; loop_height++) {
449     uint32_t color_value = start_color_val;
450     uint8_t *temp = buffer + (loop_height * buffer_stride);
451 
452     for (uint32_t loop_width = 0; loop_width < width; loop_width++) {
453       if (color_ramp == kColorRedRamp) {
454         PixelCopy(color_value, 0, 0, 0, &temp);
455         CalcCRC(color_value, &crc_red);
456         CalcCRC(0, &crc_green);
457         CalcCRC(0, &crc_blue);
458       }
459       if (color_ramp == kColorGreenRamp) {
460         PixelCopy(0, color_value, 0, 0, &temp);
461         CalcCRC(0, &crc_red);
462         CalcCRC(color_value, &crc_green);
463         CalcCRC(0, &crc_blue);
464       }
465       if (color_ramp == kColorBlueRamp) {
466         PixelCopy(0, 0, color_value, 0, &temp);
467         CalcCRC(0, &crc_red);
468         CalcCRC(0, &crc_green);
469         CalcCRC(color_value, &crc_blue);
470       }
471       if (color_ramp == kColorWhiteRamp) {
472         PixelCopy(color_value, color_value, color_value, 0, &temp);
473         CalcCRC(color_value, &crc_red);
474         CalcCRC(color_value, &crc_green);
475         CalcCRC(color_value, &crc_blue);
476       }
477 
478       color_value = (start_color_val + (((loop_width + 1) % ramp_width) * step_size)) << shift_by;
479     }
480 
481     if (panel_bpp_ == kDisplayBpp30 && ((loop_height + 1) % ramp_height) == 0) {
482       if (start_color_val == 0x180) {
483         start_color_val = 0;
484         step_size = 4;
485       } else {
486         start_color_val = 0x180;
487         step_size = 1;
488         color_ramp = (color_ramp + 1) % 4;
489       }
490       continue;
491     }
492 
493     if (((loop_height + 1) % ramp_height) == 0) {
494       color_ramp = (color_ramp + 1) % 4;
495     }
496   }
497 
498   DLOGI("CRC red %lx", crc_red.to_ulong());
499   DLOGI("CRC green %lx", crc_green.to_ulong());
500   DLOGI("CRC blue %lx", crc_blue.to_ulong());
501 }
502 
GenerateBWVertical(uint8_t * buffer)503 void HWCDisplayPluggableTest::GenerateBWVertical(uint8_t *buffer) {
504   uint32_t width = buffer_info_.buffer_config.width;
505   uint32_t height = buffer_info_.buffer_config.height;
506   LayerBufferFormat format = buffer_info_.buffer_config.format;
507   uint32_t aligned_width = buffer_info_.alloc_buffer_info.aligned_width;
508   uint32_t buffer_stride = 0;
509   uint32_t bits_per_component = panel_bpp_ / 3;
510   uint32_t max_color_val = (1 << bits_per_component) - 1;
511 
512   std::bitset<16> crc_red = {};
513   std::bitset<16> crc_green = {};
514   std::bitset<16> crc_blue = {};
515 
516   if (panel_bpp_ == kDisplayBpp18) {
517     max_color_val <<= 2;
518   }
519 
520   GetStride(format, aligned_width, &buffer_stride);
521 
522   for (uint32_t loop_height = 0; loop_height < height; loop_height++) {
523     uint32_t color = 0;
524     uint8_t *temp = buffer + (loop_height * buffer_stride);
525 
526     for (uint32_t loop_width = 0; loop_width < width; loop_width++) {
527       if (color == kColorBlack) {
528         PixelCopy(0, 0, 0, 0, &temp);
529         CalcCRC(0, &crc_red);
530         CalcCRC(0, &crc_green);
531         CalcCRC(0, &crc_blue);
532       }
533       if (color == kColorWhite) {
534         PixelCopy(max_color_val, max_color_val, max_color_val, 0, &temp);
535         CalcCRC(max_color_val, &crc_red);
536         CalcCRC(max_color_val, &crc_green);
537         CalcCRC(max_color_val, &crc_blue);
538       }
539 
540       color = (color + 1) % 2;
541     }
542   }
543 
544   DLOGI("CRC red %lx", crc_red.to_ulong());
545   DLOGI("CRC green %lx", crc_green.to_ulong());
546   DLOGI("CRC blue %lx", crc_blue.to_ulong());
547 }
548 
GenerateColorSquare(uint8_t * buffer)549 void HWCDisplayPluggableTest::GenerateColorSquare(uint8_t *buffer) {
550   uint32_t width = buffer_info_.buffer_config.width;
551   uint32_t height = buffer_info_.buffer_config.height;
552   LayerBufferFormat format = buffer_info_.buffer_config.format;
553   uint32_t aligned_width = buffer_info_.alloc_buffer_info.aligned_width;
554   uint32_t buffer_stride = 0;
555   uint32_t max_color_val = 0;
556   uint32_t min_color_val = 0;
557 
558   std::bitset<16> crc_red = {};
559   std::bitset<16> crc_green = {};
560   std::bitset<16> crc_blue = {};
561 
562   switch (panel_bpp_) {
563     case kDisplayBpp18:
564       max_color_val = 63 << 2;  // CEA Dynamic range for 18bpp 0 - 63
565       min_color_val = 0;
566       break;
567     case kDisplayBpp24:
568       max_color_val = 235;  // CEA Dynamic range for 24bpp 16 - 235
569       min_color_val = 16;
570       break;
571     case kDisplayBpp30:
572       max_color_val = 940;  // CEA Dynamic range for 30bpp 64 - 940
573       min_color_val = 64;
574       break;
575     default:
576       return;
577   }
578 
579   array<array<uint32_t, 3>, 8> colors = {{
580     {{max_color_val, max_color_val, max_color_val}},  // White Color
581     {{max_color_val, max_color_val, min_color_val}},  // Yellow Color
582     {{min_color_val, max_color_val, max_color_val}},  // Cyan Color
583     {{min_color_val, max_color_val, min_color_val}},  // Green Color
584     {{max_color_val, min_color_val, max_color_val}},  // Megenta Color
585     {{max_color_val, min_color_val, min_color_val}},  // Red Color
586     {{min_color_val, min_color_val, max_color_val}},  // Blue Color
587     {{min_color_val, min_color_val, min_color_val}},  // Black Color
588   }};
589 
590   GetStride(format, aligned_width, &buffer_stride);
591 
592   for (uint32_t loop_height = 0; loop_height < height; loop_height++) {
593     uint32_t color = 0;
594     uint8_t *temp = buffer + (loop_height * buffer_stride);
595 
596     for (uint32_t loop_width = 0; loop_width < width; loop_width++) {
597       PixelCopy(colors[color][0], colors[color][1], colors[color][2], 0, &temp);
598       CalcCRC(colors[color][0], &crc_red);
599       CalcCRC(colors[color][1], &crc_green);
600       CalcCRC(colors[color][2], &crc_blue);
601 
602       if (((loop_width + 1) % 64) == 0) {
603         color = (color + 1) % colors.size();
604       }
605     }
606 
607     if (((loop_height + 1) % 64) == 0) {
608       std::reverse(colors.begin(), (colors.end() - 1));
609     }
610   }
611 
612   DLOGI("CRC red %lx", crc_red.to_ulong());
613   DLOGI("CRC green %lx", crc_green.to_ulong());
614   DLOGI("CRC blue %lx", crc_blue.to_ulong());
615 }
616 
InitLayer(Layer * layer)617 int HWCDisplayPluggableTest::InitLayer(Layer *layer) {
618   uint32_t active_config = 0;
619   DisplayConfigVariableInfo var_info = {};
620 
621   GetActiveDisplayConfig(&active_config);
622 
623   GetDisplayAttributesForConfig(INT32(active_config), &var_info);
624 
625   layer->flags.updating = 1;
626   layer->src_rect = LayerRect(0, 0, var_info.x_pixels, var_info.y_pixels);
627   layer->dst_rect = layer->src_rect;
628   layer->frame_rate = var_info.fps;
629   layer->blending = kBlendingPremultiplied;
630 
631   layer->input_buffer.unaligned_width = var_info.x_pixels;
632   layer->input_buffer.unaligned_height = var_info.y_pixels;
633   buffer_info_.buffer_config.format = kFormatRGBA8888;
634 
635   if (layer->composition != kCompositionGPUTarget) {
636     buffer_info_.buffer_config.width = var_info.x_pixels;
637     buffer_info_.buffer_config.height = var_info.y_pixels;
638     switch (panel_bpp_) {
639       case kDisplayBpp18:
640       case kDisplayBpp24:
641         buffer_info_.buffer_config.format = kFormatRGB888;
642         break;
643       case kDisplayBpp30:
644         buffer_info_.buffer_config.format = kFormatRGBA1010102;
645         break;
646       default:
647         DLOGW("panel bpp not supported %d", panel_bpp_);
648         return -EINVAL;
649     }
650     buffer_info_.buffer_config.buffer_count = 1;
651 
652     int ret = buffer_allocator_->AllocateBuffer(&buffer_info_);
653     if (ret != 0) {
654       DLOGE("Buffer allocation failed. ret: %d", ret);
655       return -ENOMEM;
656     }
657 
658     ret = FillBuffer();
659     if (ret != 0) {
660       buffer_allocator_->FreeBuffer(&buffer_info_);
661       return ret;
662     }
663 
664     layer->input_buffer.width = buffer_info_.alloc_buffer_info.aligned_width;
665     layer->input_buffer.height = buffer_info_.alloc_buffer_info.aligned_height;
666     layer->input_buffer.size = buffer_info_.alloc_buffer_info.size;
667     layer->input_buffer.planes[0].fd = buffer_info_.alloc_buffer_info.fd;
668     layer->input_buffer.planes[0].stride = buffer_info_.alloc_buffer_info.stride;
669     layer->input_buffer.format = buffer_info_.buffer_config.format;
670 
671     DLOGI("Input buffer WxH %dx%d format %s size %d fd %d stride %d", layer->input_buffer.width,
672           layer->input_buffer.height, GetFormatString(layer->input_buffer.format),
673           layer->input_buffer.size, layer->input_buffer.planes[0].fd,
674           layer->input_buffer.planes[0].stride);
675   }
676 
677   return 0;
678 }
679 
DeinitLayer(Layer * layer)680 int HWCDisplayPluggableTest::DeinitLayer(Layer *layer) {
681   if (layer->composition != kCompositionGPUTarget) {
682     int ret = buffer_allocator_->FreeBuffer(&buffer_info_);
683     if (ret != 0) {
684       DLOGE("Buffer deallocation failed. ret: %d", ret);
685       return -ENOMEM;
686     }
687   }
688 
689   return 0;
690 }
691 
CreateLayerStack()692 int HWCDisplayPluggableTest::CreateLayerStack() {
693   for (uint32_t i = 0; i < (kTestLayerCnt + 1 /* one dummy gpu_target layer */); i++) {
694     Layer *layer = new Layer();
695 
696     if (i == kTestLayerCnt) {
697       layer->composition = kCompositionGPUTarget;
698     }
699     DLOGD("External :: CreateLayerStack %d", i);
700     int ret = InitLayer(layer);
701     if (ret != 0) {
702       delete layer;
703       return ret;
704     }
705     layer_stack_.layers.push_back(layer);
706   }
707 
708   return 0;
709 }
710 
DestroyLayerStack()711 int HWCDisplayPluggableTest::DestroyLayerStack() {
712   for (uint32_t i = 0; i < UINT32(layer_stack_.layers.size()); i++) {
713     Layer *layer = layer_stack_.layers.at(i);
714     int ret = DeinitLayer(layer);
715     if (ret != 0) {
716       return ret;
717     }
718     delete layer;
719   }
720   layer_stack_.layers = {};
721 
722   return 0;
723 }
724 
PostCommit(shared_ptr<Fence> * out_retire_fence)725 HWC2::Error HWCDisplayPluggableTest::PostCommit(shared_ptr<Fence> *out_retire_fence) {
726   auto status = HWC2::Error::None;
727   // Do no call flush on errors, if a successful buffer is never submitted.
728   if (flush_ && flush_on_error_) {
729     display_intf_->Flush(&layer_stack_);
730   }
731   flush_ = false;
732 
733   return status;
734 }
735 
736 }  // namespace sdm
737 
738