blob: e74a21fd649438b75b9a1999609a8f24176688d2 [file] [log] [blame]
Tianyu0229d252018-10-12 13:50:09 -07001#include <errno.h>
Jiwen 'Steve' Cai2e06c1c2018-07-30 21:35:32 -07002#include <private/dvr/buffer_hub_defs.h>
Fan Xu74df4902018-09-20 16:40:51 -07003#include <private/dvr/buffer_node.h>
Jiwen 'Steve' Cai2e06c1c2018-07-30 21:35:32 -07004
5namespace android {
6namespace dvr {
7
Tianyu0229d252018-10-12 13:50:09 -07008void 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' Cai9004b8c2018-10-03 18:52:23 -070018BufferNode::BufferNode(IonBuffer buffer, size_t user_metadata_size)
19 : buffer_(std::move(buffer)) {
20 metadata_ = BufferHubMetadata::Create(user_metadata_size);
Tianyu0229d252018-10-12 13:50:09 -070021 InitializeMetadata();
Jiwen 'Steve' Cai9004b8c2018-10-03 18:52:23 -070022}
Jiwen 'Steve' Cai2e06c1c2018-07-30 21:35:32 -070023
24// Allocates a new BufferNode.
25BufferNode::BufferNode(uint32_t width, uint32_t height, uint32_t layer_count,
26 uint32_t format, uint64_t usage,
Jiwen 'Steve' Cai9004b8c2018-10-03 18:52:23 -070027 size_t user_metadata_size) {
Jiwen 'Steve' Cai2e06c1c2018-07-30 21:35:32 -070028 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' Cai9004b8c2018-10-03 18:52:23 -070036 metadata_ = BufferHubMetadata::Create(user_metadata_size);
Fan Xu2cc68fb2018-10-29 13:07:53 -070037 if (!metadata_.IsValid()) {
38 ALOGE("BufferNode::BufferNode: Failed to allocate metadata.");
39 return;
40 }
Tianyu0229d252018-10-12 13:50:09 -070041 InitializeMetadata();
42}
43
44uint64_t BufferNode::GetActiveClientsBitMask() const {
45 return active_clients_bit_mask_->load(std::memory_order_acquire);
46}
47
48uint64_t BufferNode::AddNewActiveClientsBitToMask() {
49 uint64_t current_active_clients_bit_mask = GetActiveClientsBitMask();
Tianyu Jiang7e204b72018-10-26 15:39:18 -070050 uint64_t client_state_mask = 0ULL;
Tianyu0229d252018-10-12 13:50:09 -070051 uint64_t updated_active_clients_bit_mask = 0ULL;
52 do {
Tianyu Jiang7e204b72018-10-26 15:39:18 -070053 client_state_mask =
Tianyu0229d252018-10-12 13:50:09 -070054 BufferHubDefs::FindNextClearedBit(current_active_clients_bit_mask);
Tianyu Jiang7e204b72018-10-26 15:39:18 -070055 if (client_state_mask == 0ULL) {
Tianyu0229d252018-10-12 13:50:09 -070056 ALOGE(
57 "BufferNode::AddNewActiveClientsBitToMask: reached the maximum "
58 "mumber of channels per buffer node: 32.");
59 errno = E2BIG;
60 return 0ULL;
61 }
62 updated_active_clients_bit_mask =
Tianyu Jiang7e204b72018-10-26 15:39:18 -070063 current_active_clients_bit_mask | client_state_mask;
Tianyu0229d252018-10-12 13:50:09 -070064 } while (!(active_clients_bit_mask_->compare_exchange_weak(
65 current_active_clients_bit_mask, updated_active_clients_bit_mask,
66 std::memory_order_acq_rel, std::memory_order_acquire)));
Tianyu Jiang7e204b72018-10-26 15:39:18 -070067 return client_state_mask;
Tianyu0229d252018-10-12 13:50:09 -070068}
69
70void BufferNode::RemoveClientsBitFromMask(const uint64_t& value) {
71 active_clients_bit_mask_->fetch_and(~value);
Jiwen 'Steve' Cai2e06c1c2018-07-30 21:35:32 -070072}
73
74} // namespace dvr
75} // namespace android