1use crate::{
8 alloc::flags::*,
9 bindings, drm,
10 drm::driver::{AllocImpl, AllocOps},
11 error::{to_result, Result},
12 prelude::*,
13 types::{ARef, Opaque},
14};
15use core::{mem, ops::Deref, ptr, ptr::NonNull};
16
17pub trait BaseDriverObject<T: BaseObject>: Sync + Send + Sized {
19 fn new(dev: &drm::Device<T::Driver>, size: usize) -> impl PinInit<Self, Error>;
21
22 fn open(
24 _obj: &<<T as IntoGEMObject>::Driver as drm::Driver>::Object,
25 _file: &drm::File<<<T as IntoGEMObject>::Driver as drm::Driver>::File>,
26 ) -> Result {
27 Ok(())
28 }
29
30 fn close(
32 _obj: &<<T as IntoGEMObject>::Driver as drm::Driver>::Object,
33 _file: &drm::File<<<T as IntoGEMObject>::Driver as drm::Driver>::File>,
34 ) {
35 }
36}
37
38pub trait IntoGEMObject: Sized + super::private::Sealed {
40 type Driver: drm::Driver;
42
43 #[allow(clippy::wrong_self_convention)]
46 fn into_gem_obj(&self) -> &Opaque<bindings::drm_gem_object>;
47
48 fn from_gem_obj(obj: *mut bindings::drm_gem_object) -> *mut Self;
50}
51
52pub trait DriverObject: BaseDriverObject<Object<Self>> {
54 type Driver: drm::Driver;
56}
57
58extern "C" fn open_callback<T: BaseDriverObject<U>, U: BaseObject>(
59 raw_obj: *mut bindings::drm_gem_object,
60 raw_file: *mut bindings::drm_file,
61) -> core::ffi::c_int {
62 let file = unsafe {
64 drm::File::<<<U as IntoGEMObject>::Driver as drm::Driver>::File>::as_ref(raw_file)
65 };
66 let obj =
67 <<<U as IntoGEMObject>::Driver as drm::Driver>::Object as IntoGEMObject>::from_gem_obj(
68 raw_obj,
69 );
70
71 match T::open(unsafe { &*obj }, file) {
74 Err(e) => e.to_errno(),
75 Ok(()) => 0,
76 }
77}
78
79extern "C" fn close_callback<T: BaseDriverObject<U>, U: BaseObject>(
80 raw_obj: *mut bindings::drm_gem_object,
81 raw_file: *mut bindings::drm_file,
82) {
83 let file = unsafe {
85 drm::File::<<<U as IntoGEMObject>::Driver as drm::Driver>::File>::as_ref(raw_file)
86 };
87 let obj =
88 <<<U as IntoGEMObject>::Driver as drm::Driver>::Object as IntoGEMObject>::from_gem_obj(
89 raw_obj,
90 );
91
92 T::close(unsafe { &*obj }, file);
95}
96
97impl<T: DriverObject> IntoGEMObject for Object<T> {
98 type Driver = T::Driver;
99
100 fn into_gem_obj(&self) -> &Opaque<bindings::drm_gem_object> {
101 &self.obj
102 }
103
104 fn from_gem_obj(obj: *mut bindings::drm_gem_object) -> *mut Self {
105 unsafe { crate::container_of!(obj, Object<T>, obj).cast_mut() }
107 }
108}
109
110pub trait BaseObject
112where
113 Self: crate::types::AlwaysRefCounted + IntoGEMObject,
114{
115 fn size(&self) -> usize {
117 unsafe { (*self.into_gem_obj().get()).size }
120 }
121
122 fn create_handle(
125 &self,
126 file: &drm::File<<<Self as IntoGEMObject>::Driver as drm::Driver>::File>,
127 ) -> Result<u32> {
128 let mut handle: u32 = 0;
129 to_result(unsafe {
131 bindings::drm_gem_handle_create(
132 file.as_raw().cast(),
133 self.into_gem_obj().get(),
134 &mut handle,
135 )
136 })?;
137 Ok(handle)
138 }
139
140 fn lookup_handle(
142 file: &drm::File<<<Self as IntoGEMObject>::Driver as drm::Driver>::File>,
143 handle: u32,
144 ) -> Result<ARef<Self>> {
145 let ptr = unsafe { bindings::drm_gem_object_lookup(file.as_raw().cast(), handle) };
147 let ptr = <Self as IntoGEMObject>::from_gem_obj(ptr);
148 let ptr = NonNull::new(ptr).ok_or(ENOENT)?;
149
150 Ok(unsafe { ARef::from_raw(ptr) })
152 }
153
154 fn create_mmap_offset(&self) -> Result<u64> {
156 to_result(unsafe { bindings::drm_gem_create_mmap_offset(self.into_gem_obj().get()) })?;
158
159 Ok(unsafe {
161 bindings::drm_vma_node_offset_addr(ptr::addr_of_mut!(
162 (*self.into_gem_obj().get()).vma_node
163 ))
164 })
165 }
166}
167
168impl<T> BaseObject for T where Self: crate::types::AlwaysRefCounted + IntoGEMObject {}
169
170#[repr(C)]
177#[pin_data]
178pub struct Object<T: DriverObject + Send + Sync> {
179 obj: Opaque<bindings::drm_gem_object>,
180 dev: *const drm::Device<T::Driver>,
181 #[pin]
182 data: T,
183}
184
185impl<T: DriverObject> Object<T> {
186 pub const SIZE: usize = mem::size_of::<Self>();
188
189 const OBJECT_FUNCS: bindings::drm_gem_object_funcs = bindings::drm_gem_object_funcs {
190 free: Some(Self::free_callback),
191 open: Some(open_callback::<T, Object<T>>),
192 close: Some(close_callback::<T, Object<T>>),
193 print_info: None,
194 export: None,
195 pin: None,
196 unpin: None,
197 get_sg_table: None,
198 vmap: None,
199 vunmap: None,
200 mmap: None,
201 status: None,
202 vm_ops: core::ptr::null_mut(),
203 evict: None,
204 rss: None,
205 };
206
207 pub fn new(dev: &drm::Device<T::Driver>, size: usize) -> Result<ARef<Self>> {
209 let obj: Pin<KBox<Self>> = KBox::pin_init(
210 try_pin_init!(Self {
211 obj: Opaque::new(bindings::drm_gem_object::default()),
212 data <- T::new(dev, size),
213 dev,
216 }),
217 GFP_KERNEL,
218 )?;
219
220 unsafe { (*obj.as_raw()).funcs = &Self::OBJECT_FUNCS };
222
223 to_result(unsafe { bindings::drm_gem_object_init(dev.as_raw(), obj.obj.get(), size) })?;
225
226 let ptr = KBox::into_raw(unsafe { Pin::into_inner_unchecked(obj) });
228
229 let ptr = unsafe { NonNull::new_unchecked(ptr) };
231
232 Ok(unsafe { ARef::from_raw(ptr) })
234 }
235
236 pub fn dev(&self) -> &drm::Device<T::Driver> {
238 unsafe { &*self.dev }
241 }
242
243 fn as_raw(&self) -> *mut bindings::drm_gem_object {
244 self.obj.get()
245 }
246
247 extern "C" fn free_callback(obj: *mut bindings::drm_gem_object) {
248 let this = unsafe { crate::container_of!(obj, Self, obj) }.cast_mut();
250
251 unsafe { bindings::drm_gem_object_release(obj) };
254
255 let _ = unsafe { KBox::from_raw(this) };
259 }
260}
261
262unsafe impl<T: DriverObject> crate::types::AlwaysRefCounted for Object<T> {
264 fn inc_ref(&self) {
265 unsafe { bindings::drm_gem_object_get(self.as_raw()) };
267 }
268
269 unsafe fn dec_ref(obj: NonNull<Self>) {
270 let obj = unsafe { obj.as_ref() };
272
273 unsafe { bindings::drm_gem_object_put(obj.as_raw()) }
275 }
276}
277
278impl<T: DriverObject> super::private::Sealed for Object<T> {}
279
280impl<T: DriverObject> Deref for Object<T> {
281 type Target = T;
282
283 fn deref(&self) -> &Self::Target {
284 &self.data
285 }
286}
287
288impl<T: DriverObject> AllocImpl for Object<T> {
289 const ALLOC_OPS: AllocOps = AllocOps {
290 gem_create_object: None,
291 prime_handle_to_fd: None,
292 prime_fd_to_handle: None,
293 gem_prime_import: None,
294 gem_prime_import_sg_table: None,
295 dumb_create: None,
296 dumb_map_offset: None,
297 };
298}
299
300pub(super) const fn create_fops() -> bindings::file_operations {
301 let mut fops: bindings::file_operations = unsafe { core::mem::zeroed() };
304
305 fops.owner = core::ptr::null_mut();
306 fops.open = Some(bindings::drm_open);
307 fops.release = Some(bindings::drm_release);
308 fops.unlocked_ioctl = Some(bindings::drm_ioctl);
309 #[cfg(CONFIG_COMPAT)]
310 {
311 fops.compat_ioctl = Some(bindings::drm_compat_ioctl);
312 }
313 fops.poll = Some(bindings::drm_poll);
314 fops.read = Some(bindings::drm_read);
315 fops.llseek = Some(bindings::noop_llseek);
316 fops.mmap = Some(bindings::drm_gem_mmap);
317 fops.fop_flags = bindings::FOP_UNSIGNED_OFFSET;
318
319 fops
320}