1use crate::encoding::{
8 self, DynamicFlags, EpitaphBody, NoHandleResourceDialect, TransactionHeader,
9 TransactionMessage, TransactionMessageType,
10};
11use crate::error::Error;
12use crate::{AsyncChannel, Channel, TransportError};
13use zx_status;
14
15pub trait ChannelEpitaphExt {
17 fn close_with_epitaph(self, status: zx_status::Status) -> Result<(), Error>;
19}
20
21impl ChannelEpitaphExt for Channel {
22 fn close_with_epitaph(self, status: zx_status::Status) -> Result<(), Error> {
23 write_epitaph_impl(&self, status)
24 }
25}
26
27impl ChannelEpitaphExt for AsyncChannel {
28 fn close_with_epitaph(self, status: zx_status::Status) -> Result<(), Error> {
29 write_epitaph_impl(&self, status)
30 }
31}
32
33pub trait ChannelLike {
35 fn write_epitaph(&self, bytes: &[u8]) -> Result<(), TransportError>;
37}
38
39impl ChannelLike for Channel {
40 fn write_epitaph(&self, bytes: &[u8]) -> Result<(), TransportError> {
41 self.write_etc(bytes, &mut []).map_err(Into::into)
42 }
43}
44
45impl ChannelLike for AsyncChannel {
46 fn write_epitaph(&self, bytes: &[u8]) -> Result<(), TransportError> {
47 self.write_etc(bytes, &mut []).map_err(Into::into)
48 }
49}
50
51pub(crate) fn write_epitaph_impl<T: ChannelLike>(
52 channel: &T,
53 status: zx_status::Status,
54) -> Result<(), Error> {
55 let msg = TransactionMessage {
56 header: TransactionHeader::new(0, encoding::EPITAPH_ORDINAL, DynamicFlags::empty()),
57 body: &EpitaphBody { error: status },
58 };
59 encoding::with_tls_encoded::<TransactionMessageType<EpitaphBody>, NoHandleResourceDialect, ()>(
60 msg,
61 |bytes, handles| {
62 assert!(handles.is_empty(), "Epitaph should not have handles!");
63 match channel.write_epitaph(bytes) {
64 Ok(()) | Err(TransportError::Status(zx_status::Status::PEER_CLOSED)) => Ok(()),
65 Err(e) => Err(Error::ServerEpitaphWrite(e)),
66 }
67 },
68 )
69}