Tianyu | 0229d25 | 2018-10-12 13:50:09 -0700 | [diff] [blame] | 1 | #include <errno.h> |
Jiwen 'Steve' Cai | 2e06c1c | 2018-07-30 21:35:32 -0700 | [diff] [blame] | 2 | #include <private/dvr/buffer_hub_defs.h> |
Fan Xu | 74df490 | 2018-09-20 16:40:51 -0700 | [diff] [blame] | 3 | #include <private/dvr/buffer_node.h> |
Jiwen 'Steve' Cai | 2e06c1c | 2018-07-30 21:35:32 -0700 | [diff] [blame] | 4 | |
| 5 | namespace android { |
| 6 | namespace dvr { |
| 7 | |
Tianyu | 0229d25 | 2018-10-12 13:50:09 -0700 | [diff] [blame] | 8 | void BufferNode::InitializeMetadata() { |
| 9 | // Using placement new here to reuse shared memory instead of new allocation |
| 10 | // Initialize the atomic variables to zero. |
| 11 | BufferHubDefs::MetadataHeader* metadata_header = metadata_.metadata_header(); |
| 12 | buffer_state_ = new (&metadata_header->buffer_state) std::atomic<uint64_t>(0); |
| 13 | fence_state_ = new (&metadata_header->fence_state) std::atomic<uint64_t>(0); |
| 14 | active_clients_bit_mask_ = |
| 15 | new (&metadata_header->active_clients_bit_mask) std::atomic<uint64_t>(0); |
| 16 | } |
| 17 | |
Jiwen 'Steve' Cai | 9004b8c | 2018-10-03 18:52:23 -0700 | [diff] [blame] | 18 | BufferNode::BufferNode(IonBuffer buffer, size_t user_metadata_size) |
| 19 | : buffer_(std::move(buffer)) { |
| 20 | metadata_ = BufferHubMetadata::Create(user_metadata_size); |
Tianyu | 0229d25 | 2018-10-12 13:50:09 -0700 | [diff] [blame] | 21 | InitializeMetadata(); |
Jiwen 'Steve' Cai | 9004b8c | 2018-10-03 18:52:23 -0700 | [diff] [blame] | 22 | } |
Jiwen 'Steve' Cai | 2e06c1c | 2018-07-30 21:35:32 -0700 | [diff] [blame] | 23 | |
| 24 | // Allocates a new BufferNode. |
| 25 | BufferNode::BufferNode(uint32_t width, uint32_t height, uint32_t layer_count, |
| 26 | uint32_t format, uint64_t usage, |
Jiwen 'Steve' Cai | 9004b8c | 2018-10-03 18:52:23 -0700 | [diff] [blame] | 27 | size_t user_metadata_size) { |
Jiwen 'Steve' Cai | 2e06c1c | 2018-07-30 21:35:32 -0700 | [diff] [blame] | 28 | if (int ret = buffer_.Alloc(width, height, layer_count, format, usage)) { |
| 29 | ALOGE( |
| 30 | "DetachedBufferChannel::DetachedBufferChannel: Failed to allocate " |
| 31 | "buffer: %s", |
| 32 | strerror(-ret)); |
| 33 | return; |
| 34 | } |
| 35 | |
Jiwen 'Steve' Cai | 9004b8c | 2018-10-03 18:52:23 -0700 | [diff] [blame] | 36 | metadata_ = BufferHubMetadata::Create(user_metadata_size); |
Tianyu | 0229d25 | 2018-10-12 13:50:09 -0700 | [diff] [blame] | 37 | InitializeMetadata(); |
| 38 | } |
| 39 | |
| 40 | uint64_t BufferNode::GetActiveClientsBitMask() const { |
| 41 | return active_clients_bit_mask_->load(std::memory_order_acquire); |
| 42 | } |
| 43 | |
| 44 | uint64_t BufferNode::AddNewActiveClientsBitToMask() { |
| 45 | uint64_t current_active_clients_bit_mask = GetActiveClientsBitMask(); |
| 46 | uint64_t buffer_state_bit = 0ULL; |
| 47 | uint64_t updated_active_clients_bit_mask = 0ULL; |
| 48 | do { |
| 49 | buffer_state_bit = |
| 50 | BufferHubDefs::FindNextClearedBit(current_active_clients_bit_mask); |
| 51 | if (buffer_state_bit == 0ULL) { |
| 52 | ALOGE( |
| 53 | "BufferNode::AddNewActiveClientsBitToMask: reached the maximum " |
| 54 | "mumber of channels per buffer node: 32."); |
| 55 | errno = E2BIG; |
| 56 | return 0ULL; |
| 57 | } |
| 58 | updated_active_clients_bit_mask = |
| 59 | current_active_clients_bit_mask | buffer_state_bit; |
| 60 | } while (!(active_clients_bit_mask_->compare_exchange_weak( |
| 61 | current_active_clients_bit_mask, updated_active_clients_bit_mask, |
| 62 | std::memory_order_acq_rel, std::memory_order_acquire))); |
| 63 | return buffer_state_bit; |
| 64 | } |
| 65 | |
| 66 | void BufferNode::RemoveClientsBitFromMask(const uint64_t& value) { |
| 67 | active_clients_bit_mask_->fetch_and(~value); |
Jiwen 'Steve' Cai | 2e06c1c | 2018-07-30 21:35:32 -0700 | [diff] [blame] | 68 | } |
| 69 | |
| 70 | } // namespace dvr |
| 71 | } // namespace android |