xref: /DragonOS/kernel/src/driver/video/mod.rs (revision e7071df6a47c100381a8bc2000022e82d422361a)
1 use core::sync::atomic::{AtomicBool, Ordering};
2 
3 use crate::{
4     arch::MMArch,
5     driver::tty::serial::serial8250::send_to_default_serial8250_port,
6     include::bindings::bindings::{
7         FRAME_BUFFER_MAPPING_OFFSET, SPECIAL_MEMOEY_MAPPING_VIRT_ADDR_BASE,
8     },
9     init::boot_params,
10     kinfo,
11     libs::{
12         align::page_align_up,
13         lib_ui::screen_manager::{ScmBuffer, ScmBufferFlag, ScmBufferInfo},
14         rwlock::{RwLock, RwLockReadGuard},
15         spinlock::SpinLock,
16     },
17     mm::{
18         allocator::page_frame::PageFrameCount, kernel_mapper::KernelMapper, page::PageFlags,
19         MemoryManagementArch, VirtAddr,
20     },
21     time::timer::{Timer, TimerFunction},
22 };
23 use alloc::{boxed::Box, sync::Arc};
24 use system_error::SystemError;
25 
26 pub mod fbdev;
27 
28 static mut __MAMAGER: Option<VideoRefreshManager> = None;
29 
30 pub fn video_refresh_manager() -> &'static VideoRefreshManager {
31     return unsafe {
32         &__MAMAGER
33             .as_ref()
34             .expect("Video refresh manager has not been initialized yet!")
35     };
36 }
37 
38 ///管理显示刷新变量的结构体
39 pub struct VideoRefreshManager {
40     device_buffer: RwLock<ScmBufferInfo>,
41     refresh_target: RwLock<Option<Arc<SpinLock<Box<[u32]>>>>>,
42     running: AtomicBool,
43 }
44 
45 const REFRESH_INTERVAL: u64 = 30;
46 
47 impl VideoRefreshManager {
48     /**
49      * @brief 启动定时刷新
50      * @return 启动成功: true, 失败: false
51      */
52     pub fn run_video_refresh(&self) -> bool {
53         //设置Manager运行标志
54         let res = self.set_run();
55 
56         //设置成功则开始任务,否则直接返回false
57         if res {
58             //第一次将expire_jiffies设置小一点,使得这次刷新尽快开始,后续的刷新将按照REFRESH_INTERVAL间隔进行
59             let timer = Timer::new(VideoRefreshExecutor::new(), 1);
60             //将新一次定时任务加入队列
61             timer.activate();
62         }
63         return res;
64     }
65 
66     /// 停止定时刷新
67     #[allow(dead_code)]
68     pub fn stop_video_refresh(&self) {
69         self.running.store(false, Ordering::SeqCst);
70     }
71 
72     fn set_run(&self) -> bool {
73         let res = self
74             .running
75             .compare_exchange(false, true, Ordering::SeqCst, Ordering::SeqCst);
76         if res.is_ok() {
77             return true;
78         } else {
79             return false;
80         }
81     }
82 
83     /**
84      * @brief VBE帧缓存区的地址重新映射
85      * 将帧缓存区映射到地址SPECIAL_MEMOEY_MAPPING_VIRT_ADDR_BASE处
86      */
87     fn init_frame_buffer(&self) {
88         kinfo!("Re-mapping VBE frame buffer...");
89         let buf_vaddr = VirtAddr::new(
90             SPECIAL_MEMOEY_MAPPING_VIRT_ADDR_BASE as usize + FRAME_BUFFER_MAPPING_OFFSET as usize,
91         );
92         boot_params().write_irqsave().screen_info.lfb_virt_base = Some(buf_vaddr);
93 
94         let mut frame_buffer_info_graud = self.device_buffer.write();
95         if let ScmBuffer::DeviceBuffer(vaddr) = &mut (frame_buffer_info_graud).buf {
96             *vaddr = buf_vaddr;
97         }
98 
99         // 地址映射
100         let mut paddr = boot_params().read().screen_info.lfb_base;
101         let count = PageFrameCount::new(
102             page_align_up(frame_buffer_info_graud.buf_size()) / MMArch::PAGE_SIZE,
103         );
104         let page_flags: PageFlags<MMArch> = PageFlags::new().set_execute(true).set_write(true);
105 
106         let mut kernel_mapper = KernelMapper::lock();
107         let mut kernel_mapper = kernel_mapper.as_mut();
108         assert!(kernel_mapper.is_some());
109         let mut vaddr = buf_vaddr;
110         unsafe {
111             for _ in 0..count.data() {
112                 let flusher = kernel_mapper
113                     .as_mut()
114                     .unwrap()
115                     .map_phys(vaddr, paddr, page_flags)
116                     .unwrap();
117 
118                 flusher.flush();
119                 vaddr += MMArch::PAGE_SIZE;
120                 paddr += MMArch::PAGE_SIZE;
121             }
122         }
123 
124         kinfo!("VBE frame buffer successfully Re-mapped!");
125     }
126 
127     /**
128      * @brief 初始化显示模块,需先低级初始化才能高级初始化
129      * @param level 初始化等级
130      * false -> 低级初始化:不使用double buffer
131      * true ->高级初始化:增加double buffer的支持
132      * @return int
133      */
134     pub fn video_reinitialize(&self, level: bool) -> Result<(), SystemError> {
135         if !level {
136             self.init_frame_buffer();
137         } else {
138             // 开启屏幕计时刷新
139             assert!(self.run_video_refresh());
140         }
141         return Ok(());
142     }
143 
144     /**
145      * @brief 设置帧缓冲区刷新目标
146      *
147      * @param buf
148      * @return int
149      */
150     pub fn set_refresh_target(&self, buf_info: &ScmBufferInfo) -> Result<(), SystemError> {
151         let mut refresh_target = self.refresh_target.write_irqsave();
152         if let ScmBuffer::DoubleBuffer(double_buffer) = &buf_info.buf {
153             *refresh_target = Some(double_buffer.clone());
154             return Ok(());
155         }
156         return Err(SystemError::EINVAL);
157     }
158 
159     #[allow(dead_code)]
160     pub fn refresh_target(&self) -> RwLockReadGuard<'_, Option<Arc<SpinLock<Box<[u32]>>>>> {
161         let x = self.refresh_target.read();
162 
163         return x;
164     }
165 
166     pub fn device_buffer(&self) -> RwLockReadGuard<'_, ScmBufferInfo> {
167         return self.device_buffer.read();
168     }
169 
170     /// 在riscv64平台下暂时不支持
171     #[cfg(target_arch = "riscv64")]
172     pub unsafe fn video_init() -> Result<(), SystemError> {
173         return Err(SystemError::ENOSYS);
174     }
175 
176     /// 此函数用于初始化显示驱动,为后续的图形输出做好准备。
177     #[cfg(target_arch = "x86_64")]
178     pub unsafe fn video_init() -> Result<(), SystemError> {
179         use crate::{
180             arch::driver::video::arch_video_early_init,
181             driver::video::fbdev::base::BootTimeVideoType,
182         };
183 
184         arch_video_early_init()?;
185 
186         let boot_params_guard = boot_params().read();
187         let screen_info = &boot_params_guard.screen_info;
188         let buf_vaddr = screen_info.lfb_virt_base.unwrap();
189 
190         let buf_flag: ScmBufferFlag;
191         let device_buffer: ScmBufferInfo;
192 
193         if screen_info.video_type == BootTimeVideoType::Mda {
194             buf_flag = ScmBufferFlag::SCM_BF_TEXT | ScmBufferFlag::SCM_BF_FB;
195             device_buffer = ScmBufferInfo::new_device_buffer(
196                 screen_info.origin_video_cols.into(),
197                 screen_info.origin_video_lines.into(),
198                 screen_info.lfb_size as u32,
199                 screen_info.lfb_depth.into(),
200                 buf_flag,
201                 buf_vaddr,
202             )
203             .unwrap();
204         } else {
205             // 图形模式
206             buf_flag = ScmBufferFlag::SCM_BF_PIXEL | ScmBufferFlag::SCM_BF_FB;
207             device_buffer = ScmBufferInfo::new_device_buffer(
208                 screen_info.lfb_width,
209                 screen_info.lfb_height,
210                 screen_info.lfb_size as u32,
211                 screen_info.lfb_depth.into(),
212                 buf_flag,
213                 buf_vaddr,
214             )
215             .unwrap();
216         }
217 
218         let result = Self {
219             device_buffer: RwLock::new(device_buffer),
220             refresh_target: RwLock::new(None),
221             running: AtomicBool::new(false),
222         };
223 
224         __MAMAGER = Some(result);
225 
226         let init_text = "Video driver initialized.\n\0";
227         send_to_default_serial8250_port(init_text.as_bytes());
228         return Ok(());
229     }
230 }
231 
232 //刷新任务执行器
233 #[derive(Debug)]
234 struct VideoRefreshExecutor;
235 
236 impl VideoRefreshExecutor {
237     fn new() -> Box<VideoRefreshExecutor> {
238         return Box::new(VideoRefreshExecutor);
239     }
240 }
241 
242 impl TimerFunction for VideoRefreshExecutor {
243     /**
244      * @brief 交给定时器执行的任务,此方法不应手动调用
245      * @return Ok(())
246      */
247     fn run(&mut self) -> Result<(), SystemError> {
248         // 获得Manager
249         let manager = video_refresh_manager();
250 
251         let start_next_refresh = || {
252             //判断是否还需要刷新,若需要则继续分配下一次计时任务,否则不分配
253             if manager.running.load(Ordering::SeqCst) {
254                 let timer = Timer::new(VideoRefreshExecutor::new(), REFRESH_INTERVAL);
255                 //将新一次定时任务加入队列
256                 timer.activate();
257             }
258         };
259 
260         let mut refresh_target: Option<RwLockReadGuard<'_, Option<Arc<SpinLock<Box<[u32]>>>>>> =
261             None;
262         const TRY_TIMES: i32 = 2;
263         for i in 0..TRY_TIMES {
264             let g = manager.refresh_target.try_read();
265             if g.is_none() {
266                 if i == TRY_TIMES - 1 {
267                     start_next_refresh();
268                     return Ok(());
269                 }
270                 continue;
271             }
272             refresh_target = Some(g.unwrap());
273             break;
274         }
275 
276         let refresh_target = refresh_target.unwrap();
277 
278         if let ScmBuffer::DeviceBuffer(vaddr) = manager.device_buffer().buf {
279             let p = vaddr.as_ptr() as *mut u8;
280             let mut target_guard = None;
281             for _ in 0..2 {
282                 if let Ok(guard) = refresh_target.as_ref().unwrap().try_lock_irqsave() {
283                     target_guard = Some(guard);
284                     break;
285                 }
286             }
287             if target_guard.is_none() {
288                 start_next_refresh();
289                 return Ok(());
290             }
291             let mut target_guard = target_guard.unwrap();
292             unsafe {
293                 p.copy_from_nonoverlapping(
294                     target_guard.as_mut_ptr() as *mut u8,
295                     manager.device_buffer().buf_size() as usize,
296                 )
297             }
298         }
299 
300         start_next_refresh();
301 
302         return Ok(());
303     }
304 }
305 
306 #[no_mangle]
307 pub unsafe extern "C" fn rs_video_init() -> i32 {
308     return VideoRefreshManager::video_init()
309         .map(|_| 0)
310         .unwrap_or_else(|e| e.to_posix_errno());
311 }
312