summaryrefslogtreecommitdiff
path: root/src/ops/rw.rs
blob: b1c184b2b07996d96423b2a39e7986b25cbc3d49 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
use super::FromRequest;
use crate::{private_trait::Sealed, proto, util::OutputChain, Done, Operation, Reply, Request};
use std::{ffi::OsStr, os::unix::ffi::OsStrExt};

pub enum Readlink {}
pub enum Read {}
pub enum Write {}
pub enum Flush {}

pub struct WriteState {
    size: u32,
}

impl Sealed for Readlink {}
impl Sealed for Read {}
impl Sealed for Write {}
impl Sealed for Flush {}

impl<'o> Operation<'o> for Readlink {
    type RequestBody = ();
    type ReplyTail = ();
}

impl<'o> Operation<'o> for Read {
    type RequestBody = &'o proto::ReadIn;
    type ReplyTail = ();
}

impl<'o> Operation<'o> for Write {
    type RequestBody = (&'o proto::WriteIn, &'o [u8]);
    type ReplyTail = WriteState;
}

impl<'o> Operation<'o> for Flush {
    type RequestBody = &'o proto::FlushIn;
    type ReplyTail = ();
}

impl<'o> Reply<'o, Readlink> {
    /// This inode corresponds to a symbolic link pointing to the given target path.
    pub fn target<T: AsRef<OsStr>>(self, target: T) -> Done<'o> {
        self.chain(OutputChain::tail(&[target.as_ref().as_bytes()]))
    }

    /// Same as [`Reply::target()`], except that the target path is taken from disjoint
    /// slices. This involves no additional allocation.
    pub fn gather_target(self, target: &[&[u8]]) -> Done<'o> {
        self.chain(OutputChain::tail(target))
    }
}

impl<'o> Request<'o, Read> {
    pub fn handle(&self) -> u64 {
        self.body.fh
    }

    pub fn offset(&self) -> u64 {
        self.body.offset
    }

    pub fn size(&self) -> u32 {
        self.body.size
    }
}

impl<'o> Reply<'o, Read> {
    pub fn slice(self, data: &[u8]) -> Done<'o> {
        self.chain(OutputChain::tail(&[data]))
    }
}

impl<'o> Request<'o, Write> {
    pub fn handle(&self) -> u64 {
        self.body.0.fh
    }

    pub fn offset(&self) -> u64 {
        self.body.0.offset
    }

    pub fn data(&self) -> &[u8] {
        self.body.1
    }
}

impl<'o> Reply<'o, Write> {
    pub fn all(self) -> Done<'o> {
        let size = self.tail.size;
        self.single(&proto::WriteOut {
            size,
            padding: Default::default(),
        })
    }
}

impl<'o> Request<'o, Flush> {
    pub fn handle(&self) -> u64 {
        self.body.fh
    }
}

impl<'o> Reply<'o, Flush> {
    pub fn ok(self) -> Done<'o> {
        self.empty()
    }
}

impl<'o> FromRequest<'o, Write> for WriteState {
    fn from_request(request: &Request<'o, Write>) -> Self {
        let (body, data) = request.body;

        if body.size as usize != data.len() {
            log::warn!(
                "Write size={} differs from data.len={}",
                body.size,
                data.len()
            );
        }

        WriteState { size: body.size }
    }
}