mirror of
				https://github.com/RGBCube/serenity
				synced 2025-10-25 21:42:06 +00:00 
			
		
		
		
	 446200d6f3
			
		
	
	
		446200d6f3
		
	
	
	
	
		
			
			Userspace initially didn't have any sort of mechanism to handle device hotplug (either removing or inserting a device). This meant that after a short term of scanning all known devices, by fetching device events (DeviceEvent packets) from /dev/devctl, we basically never try to read it again after SystemServer initialization code. To accommodate hotplug needs, we change SystemServer by ensuring it will generate a known set of device nodes at their location during the its main initialization code. This includes devices like /dev/mem, /dev/zero and /dev/full, etc. The actual responsible userspace program to handle hotplug events is a new userspace program called DeviceMapper, with following key points: - Its current task is to to constantly read the /dev/devctl device node. Because we already created generic devices, we only handle devices that are dynamically-generated in nature, like storage devices, audio channels, etc. - Since dynamically-generated device nodes could have an infinite minor numbers, but major numbers are decoded to a device type, we create an internal registry based on two structures - DeviceNodeFamily, and RegisteredDeviceNode. DeviceNodeFamily objects are attached in the main logic code, when handling a DeviceEvent device insertion packet. A DeviceNodeFamily object has an internal HashTable to hold objects of RegisteredDeviceNode class. - Because some device nodes could still share the same major number (TTY and serial TTY devices), we have two modes of allocation - limited allocation (so a range is defined for a major number), or infinite range. Therefore, two (or more) separate DeviceNodeFamily objects can can exist albeit sharing the same major number, but they are required to allocate from a different minor numbers' range to ensure there are no collisions. - As for KCOV, we handle this device differently. In case the user compiled the kernel with such support - this happens to be a singular device node that we usually don't need, so it's dynamically-generated too, and because it has only one instance, we don't register it in our internal registry to not make it complicated needlessly. The Kernel code is modified to allow proper blocking in case of no events in the DeviceControlDevice class, because otherwise we will need to poll periodically the device to check if a new event is available, which would waste CPU time for no good reason.
		
			
				
	
	
		
			59 lines
		
	
	
	
		
			1.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			59 lines
		
	
	
	
		
			1.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2021, Liav A. <liavalb@hotmail.co.il>
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #include <Kernel/Devices/DeviceManagement.h>
 | |
| #include <Kernel/Devices/Generic/DeviceControlDevice.h>
 | |
| 
 | |
| namespace Kernel {
 | |
| 
 | |
| UNMAP_AFTER_INIT NonnullLockRefPtr<DeviceControlDevice> DeviceControlDevice::must_create()
 | |
| {
 | |
|     auto device_control_device_or_error = DeviceManagement::try_create_device<DeviceControlDevice>();
 | |
|     // FIXME: Find a way to propagate errors
 | |
|     VERIFY(!device_control_device_or_error.is_error());
 | |
|     return device_control_device_or_error.release_value();
 | |
| }
 | |
| 
 | |
| bool DeviceControlDevice::can_read(OpenFileDescription const&, u64) const
 | |
| {
 | |
|     return DeviceManagement::the().event_queue({}).with([](auto& queue) -> bool {
 | |
|         return !queue.is_empty();
 | |
|     });
 | |
| }
 | |
| 
 | |
| UNMAP_AFTER_INIT DeviceControlDevice::DeviceControlDevice()
 | |
|     : CharacterDevice(2, 10)
 | |
| {
 | |
| }
 | |
| 
 | |
| UNMAP_AFTER_INIT DeviceControlDevice::~DeviceControlDevice() = default;
 | |
| 
 | |
| ErrorOr<size_t> DeviceControlDevice::read(OpenFileDescription&, u64 offset, UserOrKernelBuffer& buffer, size_t size)
 | |
| {
 | |
|     if (offset != 0)
 | |
|         return Error::from_errno(EINVAL);
 | |
|     if ((size % sizeof(DeviceEvent)) != 0)
 | |
|         return Error::from_errno(EOVERFLOW);
 | |
| 
 | |
|     return DeviceManagement::the().event_queue({}).with([&](auto& queue) -> ErrorOr<size_t> {
 | |
|         size_t nread = 0;
 | |
|         for (size_t event_index = 0; event_index < (size / sizeof(DeviceEvent)); event_index++) {
 | |
|             if (queue.is_empty())
 | |
|                 break;
 | |
|             auto event = queue.dequeue();
 | |
|             TRY(buffer.write(&event, nread, sizeof(DeviceEvent)));
 | |
|             nread += sizeof(DeviceEvent);
 | |
|         }
 | |
|         return nread;
 | |
|     });
 | |
| }
 | |
| 
 | |
| ErrorOr<void> DeviceControlDevice::ioctl(OpenFileDescription&, unsigned, Userspace<void*>)
 | |
| {
 | |
|     return Error::from_errno(ENOTSUP);
 | |
| }
 | |
| 
 | |
| }
 |