1*61c4878aSAndroid Build Coastguard Worker // Copyright 2023 The Pigweed Authors
2*61c4878aSAndroid Build Coastguard Worker //
3*61c4878aSAndroid Build Coastguard Worker // Licensed under the Apache License, Version 2.0 (the "License"); you may not
4*61c4878aSAndroid Build Coastguard Worker // use this file except in compliance with the License. You may obtain a copy of
5*61c4878aSAndroid Build Coastguard Worker // the License at
6*61c4878aSAndroid Build Coastguard Worker //
7*61c4878aSAndroid Build Coastguard Worker // https://www.apache.org/licenses/LICENSE-2.0
8*61c4878aSAndroid Build Coastguard Worker //
9*61c4878aSAndroid Build Coastguard Worker // Unless required by applicable law or agreed to in writing, software
10*61c4878aSAndroid Build Coastguard Worker // distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
11*61c4878aSAndroid Build Coastguard Worker // WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
12*61c4878aSAndroid Build Coastguard Worker // License for the specific language governing permissions and limitations under
13*61c4878aSAndroid Build Coastguard Worker // the License.
14*61c4878aSAndroid Build Coastguard Worker
15*61c4878aSAndroid Build Coastguard Worker #include "pw_stream_shmem_mcuxpresso/stream.h"
16*61c4878aSAndroid Build Coastguard Worker
17*61c4878aSAndroid Build Coastguard Worker #include <atomic>
18*61c4878aSAndroid Build Coastguard Worker #include <cstdint>
19*61c4878aSAndroid Build Coastguard Worker
20*61c4878aSAndroid Build Coastguard Worker namespace pw::stream {
21*61c4878aSAndroid Build Coastguard Worker namespace {
22*61c4878aSAndroid Build Coastguard Worker
23*61c4878aSAndroid Build Coastguard Worker constexpr uint32_t kMuRegDataSize = 0;
24*61c4878aSAndroid Build Coastguard Worker constexpr uint32_t kMuRegDataCopied = 1;
25*61c4878aSAndroid Build Coastguard Worker
26*61c4878aSAndroid Build Coastguard Worker } // namespace
27*61c4878aSAndroid Build Coastguard Worker
~ShmemMcuxpressoStream()28*61c4878aSAndroid Build Coastguard Worker ShmemMcuxpressoStream::~ShmemMcuxpressoStream() { Disable(); }
29*61c4878aSAndroid Build Coastguard Worker
Enable()30*61c4878aSAndroid Build Coastguard Worker void ShmemMcuxpressoStream::Enable() {
31*61c4878aSAndroid Build Coastguard Worker MU_Init(base_);
32*61c4878aSAndroid Build Coastguard Worker MU_EnableInterrupts(base_,
33*61c4878aSAndroid Build Coastguard Worker kMU_Tx0EmptyInterruptEnable | kMU_Rx0FullInterruptEnable |
34*61c4878aSAndroid Build Coastguard Worker kMU_Rx1FullInterruptEnable);
35*61c4878aSAndroid Build Coastguard Worker }
36*61c4878aSAndroid Build Coastguard Worker
Disable()37*61c4878aSAndroid Build Coastguard Worker void ShmemMcuxpressoStream::Disable() {
38*61c4878aSAndroid Build Coastguard Worker MU_DisableInterrupts(base_,
39*61c4878aSAndroid Build Coastguard Worker kMU_Tx0EmptyInterruptEnable |
40*61c4878aSAndroid Build Coastguard Worker kMU_Rx0FullInterruptEnable |
41*61c4878aSAndroid Build Coastguard Worker kMU_Rx1FullInterruptEnable);
42*61c4878aSAndroid Build Coastguard Worker MU_Deinit(base_);
43*61c4878aSAndroid Build Coastguard Worker }
44*61c4878aSAndroid Build Coastguard Worker
DoRead(ByteSpan data)45*61c4878aSAndroid Build Coastguard Worker StatusWithSize ShmemMcuxpressoStream::DoRead(ByteSpan data) {
46*61c4878aSAndroid Build Coastguard Worker read_semaphore_.acquire();
47*61c4878aSAndroid Build Coastguard Worker
48*61c4878aSAndroid Build Coastguard Worker const uint32_t msg_len = MU_ReceiveMsgNonBlocking(base_, kMuRegDataSize);
49*61c4878aSAndroid Build Coastguard Worker StatusWithSize result(msg_len);
50*61c4878aSAndroid Build Coastguard Worker
51*61c4878aSAndroid Build Coastguard Worker if (msg_len > shared_read_buffer_.size()) {
52*61c4878aSAndroid Build Coastguard Worker result = StatusWithSize::Internal();
53*61c4878aSAndroid Build Coastguard Worker } else if (msg_len > data.size()) {
54*61c4878aSAndroid Build Coastguard Worker result = StatusWithSize::InvalidArgument();
55*61c4878aSAndroid Build Coastguard Worker } else {
56*61c4878aSAndroid Build Coastguard Worker std::copy(shared_read_buffer_.begin(),
57*61c4878aSAndroid Build Coastguard Worker shared_read_buffer_.begin() + msg_len,
58*61c4878aSAndroid Build Coastguard Worker data.begin());
59*61c4878aSAndroid Build Coastguard Worker // Ensure all data is read before MU message is written.
60*61c4878aSAndroid Build Coastguard Worker std::atomic_thread_fence(std::memory_order_release);
61*61c4878aSAndroid Build Coastguard Worker }
62*61c4878aSAndroid Build Coastguard Worker
63*61c4878aSAndroid Build Coastguard Worker // Ack we're done with our copy. Use blocking send as the other side will
64*61c4878aSAndroid Build Coastguard Worker // process the message directly in ISR.
65*61c4878aSAndroid Build Coastguard Worker MU_SendMsg(base_, kMuRegDataCopied, msg_len);
66*61c4878aSAndroid Build Coastguard Worker
67*61c4878aSAndroid Build Coastguard Worker // Turn back on Rx0 interrupt, which will unblock next read.
68*61c4878aSAndroid Build Coastguard Worker MU_EnableInterrupts(base_, kMU_Rx0FullInterruptEnable);
69*61c4878aSAndroid Build Coastguard Worker
70*61c4878aSAndroid Build Coastguard Worker return result;
71*61c4878aSAndroid Build Coastguard Worker }
72*61c4878aSAndroid Build Coastguard Worker
DoWrite(ConstByteSpan data)73*61c4878aSAndroid Build Coastguard Worker Status ShmemMcuxpressoStream::DoWrite(ConstByteSpan data) {
74*61c4878aSAndroid Build Coastguard Worker if (data.size() > shared_write_buffer_.size()) {
75*61c4878aSAndroid Build Coastguard Worker return Status::InvalidArgument();
76*61c4878aSAndroid Build Coastguard Worker }
77*61c4878aSAndroid Build Coastguard Worker write_semaphore_.acquire();
78*61c4878aSAndroid Build Coastguard Worker
79*61c4878aSAndroid Build Coastguard Worker std::copy(data.begin(), data.end(), shared_write_buffer_.begin());
80*61c4878aSAndroid Build Coastguard Worker
81*61c4878aSAndroid Build Coastguard Worker // Ensure MU message is written after shared buffer is populated.
82*61c4878aSAndroid Build Coastguard Worker std::atomic_thread_fence(std::memory_order_release);
83*61c4878aSAndroid Build Coastguard Worker
84*61c4878aSAndroid Build Coastguard Worker MU_SendMsgNonBlocking(base_, kMuRegDataSize, data.size());
85*61c4878aSAndroid Build Coastguard Worker
86*61c4878aSAndroid Build Coastguard Worker write_done_semaphore_.acquire();
87*61c4878aSAndroid Build Coastguard Worker
88*61c4878aSAndroid Build Coastguard Worker MU_EnableInterrupts(base_, kMU_Tx0EmptyInterruptEnable);
89*61c4878aSAndroid Build Coastguard Worker
90*61c4878aSAndroid Build Coastguard Worker return OkStatus();
91*61c4878aSAndroid Build Coastguard Worker }
92*61c4878aSAndroid Build Coastguard Worker
HandleInterrupt()93*61c4878aSAndroid Build Coastguard Worker void ShmemMcuxpressoStream::HandleInterrupt() {
94*61c4878aSAndroid Build Coastguard Worker const uint32_t flags = MU_GetInterruptsPending(base_);
95*61c4878aSAndroid Build Coastguard Worker if (flags & kMU_Tx0EmptyFlag) {
96*61c4878aSAndroid Build Coastguard Worker write_semaphore_.release();
97*61c4878aSAndroid Build Coastguard Worker MU_DisableInterrupts(base_, kMU_Tx0EmptyInterruptEnable);
98*61c4878aSAndroid Build Coastguard Worker }
99*61c4878aSAndroid Build Coastguard Worker if (flags & kMU_Rx0FullFlag) {
100*61c4878aSAndroid Build Coastguard Worker read_semaphore_.release();
101*61c4878aSAndroid Build Coastguard Worker MU_DisableInterrupts(base_, kMU_Rx0FullInterruptEnable);
102*61c4878aSAndroid Build Coastguard Worker }
103*61c4878aSAndroid Build Coastguard Worker if (flags & kMU_Rx1FullFlag) {
104*61c4878aSAndroid Build Coastguard Worker write_done_semaphore_.release();
105*61c4878aSAndroid Build Coastguard Worker MU_ReceiveMsgNonBlocking(base_, kMuRegDataCopied);
106*61c4878aSAndroid Build Coastguard Worker }
107*61c4878aSAndroid Build Coastguard Worker }
108*61c4878aSAndroid Build Coastguard Worker
109*61c4878aSAndroid Build Coastguard Worker } // namespace pw::stream
110