/* * Copyright (C) 2024 The Android Open Source Project * * 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. */ #pragma once #include #include #include #include #include #include #include #include namespace android::gui { struct DisplayLuts : public Parcelable { public: struct Entry : public Parcelable { Entry() {}; Entry(int32_t lutDimension, int32_t lutSize, int32_t lutSamplingKey) : dimension(lutDimension), size(lutSize), samplingKey(lutSamplingKey) {} int32_t dimension; int32_t size; int32_t samplingKey; status_t writeToParcel(android::Parcel* parcel) const override; status_t readFromParcel(const android::Parcel* parcel) override; }; DisplayLuts() {} DisplayLuts(base::unique_fd lutfd, std::vector lutoffsets, std::vector lutdimensions, std::vector lutsizes, std::vector lutsamplingKeys) { fd = std::move(lutfd); offsets = lutoffsets; lutProperties.reserve(offsets.size()); for (size_t i = 0; i < lutoffsets.size(); i++) { Entry entry{lutdimensions[i], lutsizes[i], lutsamplingKeys[i]}; lutProperties.emplace_back(entry); } } status_t writeToParcel(android::Parcel* parcel) const override; status_t readFromParcel(const android::Parcel* parcel) override; const base::unique_fd& getLutFileDescriptor() const { return fd; } std::vector lutProperties; std::vector offsets; private: base::unique_fd fd; }; // struct DisplayLuts static inline void PrintTo(const std::vector& offsets, ::std::ostream* os) { *os << "\n .offsets = {"; for (size_t i = 0; i < offsets.size(); i++) { *os << offsets[i]; if (i != offsets.size() - 1) { *os << ", "; } } *os << "}"; } static inline void PrintTo(const std::vector& entries, ::std::ostream* os) { *os << "\n .lutProperties = {\n"; for (auto& [dimension, size, samplingKey] : entries) { *os << " Entry{" << "dimension: " << dimension << ", size: " << size << ", samplingKey: " << samplingKey << "}\n"; } *os << " }"; } static constexpr size_t kMaxPrintCount = 100; static inline void PrintTo(const std::vector& buffer, size_t offset, int32_t dimension, size_t size, ::std::ostream* os) { size_t range = std::min(size, kMaxPrintCount); *os << "{"; if (dimension == 1) { for (size_t i = 0; i < range; i++) { *os << buffer[offset + i]; if (i != range - 1) { *os << ", "; } } } else { *os << "\n {R channel:"; for (size_t i = 0; i < range; i++) { *os << buffer[offset + i]; if (i != range - 1) { *os << ", "; } } *os << "}\n {G channel:"; for (size_t i = 0; i < range; i++) { *os << buffer[offset + size + i]; if (i != range - 1) { *os << ", "; } } *os << "}\n {B channel:"; for (size_t i = 0; i < range; i++) { *os << buffer[offset + 2 * size + i]; if (i != range - 1) { *os << ", "; } } } *os << "}"; } static inline void PrintTo(const std::shared_ptr luts, ::std::ostream* os) { *os << "gui::DisplayLuts {"; auto& fd = luts->getLutFileDescriptor(); *os << "\n .pfd = " << fd.get(); if (fd.ok()) { PrintTo(luts->offsets, os); PrintTo(luts->lutProperties, os); // decode luts int32_t fullLength = luts->offsets[luts->offsets.size() - 1]; if (luts->lutProperties[luts->offsets.size() - 1].dimension == 1) { fullLength += luts->lutProperties[luts->offsets.size() - 1].size; } else { fullLength += (luts->lutProperties[luts->offsets.size() - 1].size * luts->lutProperties[luts->offsets.size() - 1].size * luts->lutProperties[luts->offsets.size() - 1].size * 3); } size_t bufferSize = static_cast(fullLength) * sizeof(float); float* ptr = (float*)mmap(NULL, bufferSize, PROT_READ, MAP_SHARED, fd.get(), 0); if (ptr == MAP_FAILED) { *os << "\n .bufferdata cannot mmap!"; return; } std::vector buffers(ptr, ptr + fullLength); munmap(ptr, bufferSize); *os << "\n .bufferdata = "; for (size_t i = 0; i < luts->offsets.size(); i++) { PrintTo(buffers, static_cast(luts->offsets[i]), luts->lutProperties[i].dimension, static_cast(luts->lutProperties[i].size), os); } } *os << "\n }"; } } // namespace android::gui