cryptoki 0.12.0

Rust-native wrapper around the PKCS #11 API
Documentation
// Copyright 2021 Contributors to the Parsec project.
// SPDX-License-Identifier: Apache-2.0
//! Session management functions

use cryptoki_sys::{CKF_RW_SESSION, CKF_SERIAL_SESSION};

use crate::context::Pkcs11;
use crate::error::{Result, Rv};
use crate::session::{CloseOnDrop, Session};
use crate::slot::Slot;

use super::Function;

impl Pkcs11 {
    #[inline(always)]
    fn open_session(
        &self,
        slot_id: Slot,
        read_write: bool,
        close_on_drop: CloseOnDrop,
    ) -> Result<Session> {
        let mut session_handle = 0;

        let flags = if read_write {
            CKF_SERIAL_SESSION | CKF_RW_SESSION
        } else {
            CKF_SERIAL_SESSION
        };
        unsafe {
            Rv::from(get_pkcs11!(self, C_OpenSession)(
                slot_id.into(),
                flags,
                // TODO: abstract those types or create new functions for callbacks
                std::ptr::null_mut(),
                None,
                &mut session_handle,
            ))
            .into_result(Function::OpenSession)?;
        }

        Ok(Session::new(session_handle, self.clone(), close_on_drop))
    }

    /// Open a new Read-Only session
    ///
    /// For a Read-Write session, use `open_rw_session`
    ///
    /// Note: No callback is set when opening the session.
    ///
    /// # Examples
    ///
    /// ```rust
    /// # fn main() -> testresult::TestResult {
    /// use cryptoki::session::Session;
    /// use cryptoki::context::Pkcs11;
    /// use cryptoki::context::{CInitializeArgs, CInitializeFlags};
    ///
    ///
    /// let mut client = Pkcs11::new(
    ///     std::env::var("TEST_PKCS11_MODULE")
    ///        .unwrap_or_else(|_| "/usr/local/lib/softhsm/libsofthsm2.so".to_string()),
    /// )?;
    /// client.initialize(CInitializeArgs::new(CInitializeFlags::OS_LOCKING_OK))?;
    ///
    /// // Use the first slot
    /// let slot = client.get_all_slots()?[0];
    /// let session = client.open_ro_session(slot)?;
    /// # let _ = session; Ok(()) }
    /// ```
    pub fn open_ro_session(&self, slot_id: Slot) -> Result<Session> {
        self.open_session(slot_id, false, CloseOnDrop::AutomaticallyCloseSession)
    }

    /// Open a new Read/Write session
    ///
    /// Note: No callback is set when opening the session.
    pub fn open_rw_session(&self, slot_id: Slot) -> Result<Session> {
        self.open_session(slot_id, true, CloseOnDrop::AutomaticallyCloseSession)
    }

    /// Open a new Read-Only session without automatic close on drop.
    ///
    /// The caller is responsible for calling `close()` explicitly.
    ///
    /// Note: No callback is set when opening the session.
    pub fn open_ro_session_no_drop(&self, slot_id: Slot) -> Result<Session> {
        self.open_session(slot_id, false, CloseOnDrop::DoNotClose)
    }

    /// Open a new Read/Write session without automatic close on drop.
    ///
    /// The caller is responsible for calling `close()` explicitly.
    ///
    /// Note: No callback is set when opening the session.
    pub fn open_rw_session_no_drop(&self, slot_id: Slot) -> Result<Session> {
        self.open_session(slot_id, true, CloseOnDrop::DoNotClose)
    }
}