1use crate::prelude_internal::*;
8use num_derive::FromPrimitive;
9
10#[derive(Debug, Copy, Clone, Eq, Ord, PartialOrd, PartialEq, FromPrimitive)]
13pub enum Priority {
14 Low = OT_MESSAGE_PRIORITY_LOW as isize,
17
18 Normal = OT_MESSAGE_PRIORITY_NORMAL as isize,
23
24 High = OT_MESSAGE_PRIORITY_HIGH as isize,
27}
28
29impl Default for Priority {
30 fn default() -> Self {
32 Priority::Normal
33 }
34}
35
36impl Priority {
37 pub fn as_ot_message_priority(self) -> otMessagePriority {
39 self as otMessagePriority
40 }
41}
42
43impl From<otMessagePriority> for Priority {
44 fn from(x: otMessagePriority) -> Self {
45 use num::FromPrimitive;
46 Self::from_u32(x).unwrap_or_else(|| panic!("Unknown otMessagePriority value: {x}"))
47 }
48}
49
50impl From<Priority> for otMessagePriority {
51 fn from(x: Priority) -> Self {
52 x.as_ot_message_priority()
53 }
54}
55
56impl From<u8> for Priority {
57 fn from(x: u8) -> Self {
58 use num::FromPrimitive;
59 Self::from_u8(x).unwrap_or_else(|| panic!("Unknown otMessagePriority value: {x}"))
60 }
61}
62
63impl From<Priority> for u8 {
64 #[allow(clippy::unnecessary_fallible_conversions)]
65 fn from(x: Priority) -> Self {
66 x.try_into().unwrap()
67 }
68}
69
70#[derive(Clone)]
73#[repr(transparent)]
74pub struct Info(pub otMessageInfo);
75
76impl_ot_castable!(Info, otMessageInfo);
77
78impl AsRef<Info> for Info {
79 fn as_ref(&self) -> &Info {
80 self
81 }
82}
83
84impl Info {
85 pub fn new(local: SockAddr, remote: SockAddr) -> Info {
87 Info(otMessageInfo {
88 mSockAddr: local.addr().into_ot(),
89 mPeerAddr: remote.addr().into_ot(),
90 mSockPort: local.port(),
91 mPeerPort: remote.port(),
92 ..otMessageInfo::default()
93 })
94 }
95
96 pub fn ecn(&self) -> u8 {
98 self.0.mEcn()
99 }
100
101 pub fn set_ecn(&mut self, ecn: u8) {
103 self.0.set_mEcn(ecn)
104 }
105
106 pub fn hop_limit(&self) -> u8 {
108 self.0.mHopLimit
109 }
110
111 pub fn set_hop_limit(&mut self, hop_limit: u8) {
113 self.0.mHopLimit = hop_limit
114 }
115
116 pub fn is_host_interface(&self) -> bool {
118 self.0.mIsHostInterface()
119 }
120
121 pub fn set_host_interface(&mut self, host_interface: bool) {
123 self.0.set_mIsHostInterface(host_interface)
124 }
125
126 pub fn multicast_loop(&self) -> bool {
128 self.0.mMulticastLoop()
129 }
130
131 pub fn set_multicast_loop(&mut self, multicast_loop: bool) {
133 self.0.set_mMulticastLoop(multicast_loop)
134 }
135
136 pub fn allow_zero_hop_limit(&self) -> bool {
138 self.0.mAllowZeroHopLimit()
139 }
140
141 pub fn set_allow_zero_hop_limit(&mut self, allow_zero_hop_limit: bool) {
143 self.0.set_mAllowZeroHopLimit(allow_zero_hop_limit)
144 }
145
146 pub fn sock_name(&self) -> SockAddr {
148 SockAddr::new(Ip6Address::from_ot(self.0.mSockAddr), self.0.mSockPort)
149 }
150
151 pub fn peer_name(&self) -> SockAddr {
153 SockAddr::new(Ip6Address::from_ot(self.0.mPeerAddr), self.0.mPeerPort)
154 }
155}
156
157impl std::fmt::Debug for Info {
158 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
159 f.debug_struct("otMessageInfo")
160 .field("sock_name", &self.sock_name())
161 .field("peer_name", &self.peer_name())
162 .field("ecn", &self.ecn())
163 .field("hop_limit", &self.hop_limit())
164 .field("allow_zero_hop_limit", &self.allow_zero_hop_limit())
165 .field("multicast_loop", &self.multicast_loop())
166 .field("is_host_interface", &self.is_host_interface())
167 .finish()
168 }
169}
170
171#[derive(Debug, Clone)]
174#[repr(transparent)]
175pub struct Settings(pub otMessageSettings);
176
177impl_ot_castable!(Settings, otMessageSettings);
178
179impl AsRef<Settings> for Settings {
180 fn as_ref(&self) -> &Settings {
181 self
182 }
183}
184
185impl Default for Settings {
186 fn default() -> Self {
187 Settings(otMessageSettings {
188 mLinkSecurityEnabled: true,
189 mPriority: OT_MESSAGE_PRIORITY_NORMAL.try_into().unwrap(),
190 })
191 }
192}
193
194impl Settings {
195 pub fn set_priority(mut self, priority: Priority) -> Self {
197 self.0.mPriority = priority.into();
198 self
199 }
200
201 pub fn set_link_security(mut self, enabled: bool) -> Self {
203 self.0.mLinkSecurityEnabled = enabled;
204 self
205 }
206}
207
208#[repr(transparent)]
217pub struct Message<'a>(otMessage, PhantomData<*mut otMessage>, PhantomData<&'a ()>);
218
219impl std::fmt::Debug for Message<'_> {
220 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
221 f.debug_struct("otMessage")
222 .field("otPtr", &self.as_ot_ptr())
223 .field("len", &self.len())
224 .field("offset", &self.offset())
225 .field("RSSI", &self.rssi())
226 .finish()
227 }
228}
229
230unsafe impl ot::Boxable for Message<'_> {
233 type OtType = otMessage;
234 unsafe fn finalize(&mut self) {
235 otMessageFree(self.as_ot_ptr())
236 }
237}
238
239impl<'a> Message<'a> {
240 pub fn ip6_new<T: ot::Boxable<OtType = otInstance>>(
242 instance: &'a T,
243 settings: Option<&Settings>,
244 ) -> Result<ot::Box<Message<'a>>, ot::NoBufs> {
245 unsafe {
246 ot::Box::from_ot_ptr(otIp6NewMessage(
250 instance.as_ot_ptr(),
251 settings.map(|x| x.as_ref().as_ot_ptr()).unwrap_or(null()),
252 ))
253 }
254 .ok_or(ot::NoBufs)
255 }
256
257 pub fn ip6_new_from_bytes<T: ot::Boxable<OtType = otInstance>, D: AsRef<[u8]>>(
264 instance: &'a T,
265 data: D,
266 settings: Option<&Settings>,
267 ) -> Result<ot::Box<Message<'a>>, ot::MalformedOrNoBufs> {
268 let data = data.as_ref();
269 unsafe {
270 ot::Box::from_ot_ptr(otIp6NewMessageFromBuffer(
274 instance.as_ot_ptr(),
275 data.as_ptr(),
276 data.len().try_into().expect("ot::Message::ip6_new_from_bytes: Buffer too large"),
277 settings.map(|x| x.as_ref().as_ot_ptr()).unwrap_or(null()),
278 ))
279 }
280 .ok_or(ot::MalformedOrNoBufs)
281 }
282
283 pub fn ip4_new<T: ot::Boxable<OtType = otInstance>>(
285 instance: &'a T,
286 settings: Option<&Settings>,
287 ) -> Result<ot::Box<Message<'a>>, ot::NoBufs> {
288 unsafe {
289 ot::Box::from_ot_ptr(otIp4NewMessage(
293 instance.as_ot_ptr(),
294 settings.map(|x| x.as_ref().as_ot_ptr()).unwrap_or(null()),
295 ))
296 }
297 .ok_or(ot::NoBufs)
298 }
299}
300
301impl<'a> Message<'a> {
302 pub fn udp_new<T: ot::Boxable<OtType = otInstance>>(
304 instance: &'a T,
305 settings: Option<&Settings>,
306 ) -> Result<ot::Box<Message<'a>>, ot::NoBufs> {
307 unsafe {
308 ot::Box::from_ot_ptr(otUdpNewMessage(
312 instance.as_ot_ptr(),
313 settings.map(|x| x.as_ref().as_ot_ptr()).unwrap_or(null()),
314 ))
315 }
316 .ok_or(ot::NoBufs)
317 }
318}
319
320impl Message<'_> {
321 pub fn append(&mut self, data: &[u8]) -> Result {
326 Error::from(unsafe {
327 otMessageAppend(
330 self.as_ot_ptr(),
331 data.as_ptr() as *const ::std::os::raw::c_void,
332 data.len().try_into().or(Err(Error::InvalidArgs))?,
333 )
334 })
335 .into()
336 }
337
338 pub fn is_link_security_enabled(&self) -> bool {
341 unsafe { otMessageIsLinkSecurityEnabled(self.as_ot_ptr()) }
342 }
343
344 #[allow(clippy::len_without_is_empty)]
346 pub fn len(&self) -> usize {
347 unsafe { otMessageGetLength(self.as_ot_ptr()).into() }
348 }
349
350 pub fn set_len(&mut self, len: usize) -> Result {
355 Error::from(unsafe {
356 otMessageSetLength(self.as_ot_ptr(), len.try_into().or(Err(Error::InvalidArgs))?)
359 })
360 .into()
361 }
362
363 pub fn offset(&self) -> usize {
365 unsafe { otMessageGetOffset(self.as_ot_ptr()).into() }
366 }
367
368 pub fn set_offset(&mut self, offset: usize) {
372 unsafe {
373 otMessageSetOffset(
376 self.as_ot_ptr(),
377 offset.try_into().expect("ot::Message::set_offset: Offset too large"),
378 )
379 }
380 }
381
382 pub fn set_direct_transmission(&mut self, direct_transmission: bool) {
388 unsafe {
389 otMessageSetDirectTransmission(self.as_ot_ptr(), direct_transmission);
390 }
391 }
392
393 pub fn rssi(&self) -> Decibels {
395 unsafe { otMessageGetRss(self.as_ot_ptr()) }
396 }
397
398 pub fn to_vec(&self) -> Vec<u8> {
401 let mut buffer = Vec::with_capacity(self.len());
402 #[allow(clippy::uninit_vec)] unsafe {
404 buffer.set_len(self.len());
405 otMessageRead(
406 self.as_ot_ptr(),
407 0,
408 buffer.as_mut_ptr() as *mut core::ffi::c_void,
409 otMessageGetLength(self.as_ot_ptr()),
410 );
411 }
412 buffer
413 }
414}
415
416#[derive(Debug, Default, Copy, Clone)]
419#[repr(transparent)]
420pub struct BufferInfo(pub otBufferInfo);
421
422impl_ot_castable!(BufferInfo, otBufferInfo);
423
424pub trait MessageBuffer {
426 fn get_buffer_info(&self) -> BufferInfo;
428}
429
430impl<T: MessageBuffer + Boxable> MessageBuffer for ot::Box<T> {
431 fn get_buffer_info(&self) -> BufferInfo {
432 self.as_ref().get_buffer_info()
433 }
434}
435
436impl MessageBuffer for Instance {
437 fn get_buffer_info(&self) -> BufferInfo {
438 let mut ret = BufferInfo::default();
439 unsafe {
440 otMessageGetBufferInfo(self.as_ot_ptr(), ret.as_ot_mut_ptr());
441 }
442 ret
443 }
444}