xref: /aosp_15_r20/external/pigweed/pw_stream_shmem_mcuxpresso/stream.cc (revision 61c4878ac05f98d0ceed94b57d316916de578985)
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