aboutsummaryrefslogtreecommitdiff
path: root/src/bio/mod.rs
blob: f81114de614a8478e9490f073a908c1deb2fa210 (plain) (blame)
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
use libc::{c_void, c_int};
use std::io::{IoResult, IoError, OtherIoError};
use std::io::{Reader, Writer};
use std::ptr;

use ssl::error::{SslError};

pub struct MemBio {
    bio: *mut ffi::BIO,
    owned: bool
}

impl Drop for MemBio {
    fn drop(&mut self) {
        if self.owned {
            unsafe {
                ffi::BIO_free_all(self.bio);
            }
        }
    }
}

impl MemBio {
    /// Creates a new owned memory based BIO
    pub fn new() -> Result<MemBio, SslError> {
        let bio = unsafe { ffi::BIO_new(ffi::BIO_s_mem()) };
        try_ssl_null!(bio);

        Ok(MemBio {
            bio: bio,
            owned: true
        })
    }

    /// Returns a "borrow", i.e. it has no ownership
    pub fn borrowed(bio: *mut ffi::BIO) -> MemBio {
        MemBio {
            bio: bio,
            owned: false
        }
    }

    /// Consumes current bio and returns wrapped value
    /// Note that data ownership is lost and
    /// should be handled manually
    pub unsafe fn unwrap(mut self) -> *mut ffi::BIO {
        self.owned = false;
        self.bio
    }

    /// Temporarily gets wrapped value
    pub unsafe fn get_handle(&self) -> *mut ffi::BIO {
        self.bio
    }
}

impl Reader for MemBio {
    fn read(&mut self, buf: &mut [u8]) -> IoResult<uint> {
        let ret = unsafe {
            ffi::BIO_read(self.bio, buf.as_ptr() as *mut c_void,
                          buf.len() as c_int)
        };

        if ret < 0 {
            // FIXME: provide details from OpenSSL
            Err(IoError{kind: OtherIoError, desc: "mem bio read error", detail: None})
        } else {
            Ok(ret as uint)
        }
    }
}

impl Writer for MemBio {
    fn write(&mut self, buf: &[u8]) -> IoResult<()> {
        let ret = unsafe {
            ffi::BIO_write(self.bio, buf.as_ptr() as *const c_void,
                           buf.len() as c_int)
        };
        if buf.len() != ret as uint {
            // FIXME: provide details from OpenSSL
            Err(IoError{kind: OtherIoError, desc: "mem bio write error", detail: None})
        } else {
            Ok(())
        }
    }
}

pub mod ffi {
    #![allow(non_camel_case_types)]

    use libc::{c_int, c_void};

    pub type BIO = c_void;
    pub type BIO_METHOD = c_void;

    extern "C" {
        pub fn BIO_s_mem() -> *const BIO_METHOD;
        pub fn BIO_new(type_: *const BIO_METHOD) -> *mut BIO;
        pub fn BIO_free_all(a: *mut BIO);
        pub fn BIO_read(b: *mut BIO, buf: *mut c_void, len: c_int) -> c_int;
        pub fn BIO_write(b: *mut BIO, buf: *const c_void, len: c_int) -> c_int;
    }
}