fidl/
epitaph.rs

1// Copyright 2019 The Fuchsia Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5//! Epitaph support for Channel and AsyncChannel.
6
7use 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
15/// Extension trait that provides Channel-like objects with the ability to send a FIDL epitaph.
16pub trait ChannelEpitaphExt {
17    /// Consumes the channel and writes an epitaph.
18    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
33/// Indicates an object is "channel-like" in that it can receive epitaphs.
34pub trait ChannelLike {
35    /// Write an epitaph to a channel. Same as write_etc but is never fed handles.
36    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}