unplugged-system/external/crosvm/media/libva/src/context.rs

96 lines
2.6 KiB
Rust

// Copyright 2022 The ChromiumOS Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
use std::rc::Rc;
use anyhow::Result;
use log::error;
use crate::bindings;
use crate::buffer::Buffer;
use crate::buffer_type::BufferType;
use crate::display::Display;
use crate::status::Status;
use crate::Config;
use crate::Surface;
/// A VA context for a particular [`Display`].
pub struct Context {
display: Rc<Display>,
id: bindings::VAContextID,
}
impl Context {
/// Creates a Context by wrapping around a `vaCreateContext` call. This is just a helper for
/// [`Display::create_context`].
pub(crate) fn new(
display: Rc<Display>,
config: &Config,
coded_width: i32,
coded_height: i32,
surfaces: Option<&Vec<Surface>>,
progressive: bool,
) -> Result<Rc<Self>> {
let mut context_id = 0;
let flags = if progressive {
bindings::constants::VA_PROGRESSIVE as i32
} else {
0
};
let mut render_targets = match surfaces {
Some(surfaces) => Surface::as_id_vec(surfaces),
None => Default::default(),
};
// Safe because `self` represents a valid VADisplay and render_targets
// and ntargets are properly initialized. Note that render_targets==NULL
// is valid so long as ntargets==0.
Status(unsafe {
bindings::vaCreateContext(
display.handle(),
config.id(),
coded_width,
coded_height,
flags,
render_targets.as_mut_ptr(),
render_targets.len() as i32,
&mut context_id,
)
})
.check()?;
Ok(Rc::new(Self {
display,
id: context_id,
}))
}
/// Returns a shared reference to the [`Display`] used by this context.
pub fn display(&self) -> Rc<Display> {
Rc::clone(&self.display)
}
/// Returns the ID of this context.
pub(crate) fn id(&self) -> bindings::VAContextID {
self.id
}
/// Create a new buffer of type `type_`.
pub fn create_buffer(self: &Rc<Self>, type_: BufferType) -> Result<Buffer> {
Buffer::new(Rc::clone(self), type_)
}
}
impl Drop for Context {
fn drop(&mut self) {
// Safe because `self` represents a valid VAContext.
let status =
Status(unsafe { bindings::vaDestroyContext(self.display.handle(), self.id) }).check();
if status.is_err() {
error!("vaDestroyContext failed: {}", status.unwrap_err());
}
}
}