Console.cpp 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218
  1. /*
  2. * Copyright (c) 2021, the SerenityOS developers.
  3. * Copyright (c) 2021, Kyle Pereira <hey@xylepereira.me>
  4. *
  5. * SPDX-License-Identifier: BSD-2-Clause
  6. */
  7. #include <Kernel/Bus/VirtIO/Console.h>
  8. #include <Kernel/Sections.h>
  9. namespace Kernel::VirtIO {
  10. unsigned Console::next_device_id = 0;
  11. UNMAP_AFTER_INIT Console::Console(PCI::Address address)
  12. : VirtIO::Device(address)
  13. , m_device_id(next_device_id++)
  14. {
  15. if (auto cfg = get_config(ConfigurationType::Device)) {
  16. bool success = negotiate_features([&](u64 supported_features) {
  17. u64 negotiated = 0;
  18. if (is_feature_set(supported_features, VIRTIO_CONSOLE_F_SIZE))
  19. dbgln("VirtIO::Console: Console size is not yet supported!");
  20. if (is_feature_set(supported_features, VIRTIO_CONSOLE_F_MULTIPORT))
  21. negotiated |= VIRTIO_CONSOLE_F_MULTIPORT;
  22. return negotiated;
  23. });
  24. if (success) {
  25. u32 max_nr_ports = 0;
  26. u16 cols = 0, rows = 0;
  27. read_config_atomic([&]() {
  28. if (is_feature_accepted(VIRTIO_CONSOLE_F_SIZE)) {
  29. cols = config_read16(*cfg, 0x0);
  30. rows = config_read16(*cfg, 0x2);
  31. }
  32. if (is_feature_accepted(VIRTIO_CONSOLE_F_MULTIPORT)) {
  33. max_nr_ports = config_read32(*cfg, 0x4);
  34. m_ports.resize(max_nr_ports);
  35. }
  36. });
  37. dbgln("VirtIO::Console: cols: {}, rows: {}, max nr ports {}", cols, rows, max_nr_ports);
  38. // Base receiveq/transmitq for port0 + optional control queues and 2 per every additional port
  39. success = setup_queues(2 + max_nr_ports > 0 ? 2 + 2 * max_nr_ports : 0);
  40. }
  41. if (success) {
  42. finish_init();
  43. if (is_feature_accepted(VIRTIO_CONSOLE_F_MULTIPORT))
  44. setup_multiport();
  45. else
  46. m_ports.append(new VirtIO::ConsolePort(0u, *this));
  47. }
  48. }
  49. }
  50. bool Console::handle_device_config_change()
  51. {
  52. dbgln("VirtIO::Console: Handle device config change");
  53. return true;
  54. }
  55. void Console::handle_queue_update(u16 queue_index)
  56. {
  57. dbgln_if(VIRTIO_DEBUG, "VirtIO::Console: Handle queue update {}", queue_index);
  58. if (queue_index == CONTROL_RECEIVEQ) {
  59. SpinlockLocker ringbuffer_lock(m_control_receive_buffer->lock());
  60. auto& queue = get_queue(CONTROL_RECEIVEQ);
  61. SpinlockLocker queue_lock(queue.lock());
  62. size_t used;
  63. QueueChain popped_chain = queue.pop_used_buffer_chain(used);
  64. while (!popped_chain.is_empty()) {
  65. popped_chain.for_each([&](auto addr, auto) {
  66. auto offset = addr.as_ptr() - m_control_receive_buffer->start_of_region().as_ptr();
  67. auto* message = reinterpret_cast<ControlMessage*>(m_control_receive_buffer->vaddr().offset(offset).as_ptr());
  68. process_control_message(*message);
  69. });
  70. supply_chain_and_notify(CONTROL_RECEIVEQ, popped_chain);
  71. popped_chain = queue.pop_used_buffer_chain(used);
  72. }
  73. } else if (queue_index == CONTROL_TRANSMITQ) {
  74. SpinlockLocker ringbuffer_lock(m_control_transmit_buffer->lock());
  75. auto& queue = get_queue(CONTROL_TRANSMITQ);
  76. SpinlockLocker queue_lock(queue.lock());
  77. size_t used;
  78. QueueChain popped_chain = queue.pop_used_buffer_chain(used);
  79. auto number_of_messages = 0;
  80. do {
  81. popped_chain.for_each([this](PhysicalAddress address, size_t length) {
  82. m_control_transmit_buffer->reclaim_space(address, length);
  83. });
  84. popped_chain.release_buffer_slots_to_queue();
  85. popped_chain = queue.pop_used_buffer_chain(used);
  86. number_of_messages++;
  87. } while (!popped_chain.is_empty());
  88. m_control_wait_queue.wake_n(number_of_messages);
  89. } else {
  90. u32 port_index = queue_index < 2 ? 0 : (queue_index - 2) / 2;
  91. if (port_index >= m_ports.size() || !m_ports.at(port_index)) {
  92. dbgln("Invalid queue_index {}", queue_index);
  93. return;
  94. }
  95. m_ports.at(port_index)->handle_queue_update({}, queue_index);
  96. }
  97. }
  98. void Console::setup_multiport()
  99. {
  100. m_control_receive_buffer = make<Memory::RingBuffer>("VirtIOConsole control receive queue", CONTROL_BUFFER_SIZE);
  101. m_control_transmit_buffer = make<Memory::RingBuffer>("VirtIOConsole control transmit queue", CONTROL_BUFFER_SIZE);
  102. auto& queue = get_queue(CONTROL_RECEIVEQ);
  103. SpinlockLocker queue_lock(queue.lock());
  104. QueueChain chain(queue);
  105. auto offset = 0ul;
  106. while (offset < CONTROL_BUFFER_SIZE) {
  107. auto buffer_start = m_control_receive_buffer->start_of_region().offset(offset);
  108. auto did_add_buffer = chain.add_buffer_to_chain(buffer_start, CONTROL_MESSAGE_SIZE, BufferType::DeviceWritable);
  109. VERIFY(did_add_buffer);
  110. offset += CONTROL_MESSAGE_SIZE;
  111. supply_chain_and_notify(CONTROL_RECEIVEQ, chain);
  112. }
  113. ControlMessage ready_event {
  114. .id = 0, // Unused
  115. .event = (u16)ControlEvent::DeviceReady,
  116. .value = (u16)ControlMessage::Status::Success
  117. };
  118. write_control_message(ready_event);
  119. }
  120. void Console::process_control_message(ControlMessage message)
  121. {
  122. switch (message.event) {
  123. case (u16)ControlEvent::DeviceAdd: {
  124. u32 id = message.id;
  125. if (id >= m_ports.size()) {
  126. dbgln("Device provided an invalid port number {}. max_nr_ports: {}", id, m_ports.size());
  127. return;
  128. } else if (!m_ports.at(id).is_null()) {
  129. dbgln("Device tried to add port {} which was already added!", id);
  130. return;
  131. }
  132. m_ports.at(id) = new VirtIO::ConsolePort(id, *this);
  133. ControlMessage ready_event {
  134. .id = static_cast<u32>(id),
  135. .event = (u16)ControlEvent::PortReady,
  136. .value = (u16)ControlMessage::Status::Success
  137. };
  138. write_control_message(ready_event);
  139. break;
  140. }
  141. case (u16)ControlEvent::ConsolePort:
  142. case (u16)ControlEvent::PortOpen: {
  143. if (message.id >= m_ports.size()) {
  144. dbgln("Device provided an invalid port number {}. max_nr_ports: {}", message.id, m_ports.size());
  145. return;
  146. } else if (m_ports.at(message.id).is_null()) {
  147. dbgln("Device tried to open port {} which was not added!", message.id);
  148. return;
  149. }
  150. if (message.value == (u16)ControlMessage::PortStatus::Open) {
  151. auto is_open = m_ports.at(message.id)->is_open();
  152. if (!is_open) {
  153. m_ports.at(message.id)->set_open({}, true);
  154. send_open_control_message(message.id, true);
  155. }
  156. } else if (message.value == (u16)ControlMessage::PortStatus::Close) {
  157. m_ports.at(message.id)->set_open({}, false);
  158. } else {
  159. dbgln("Device specified invalid value {}. Must be 0 or 1.", message.value);
  160. }
  161. break;
  162. }
  163. default:
  164. dbgln("Unhandled message event {}!", message.event);
  165. }
  166. }
  167. void Console::write_control_message(ControlMessage message)
  168. {
  169. SpinlockLocker ringbuffer_lock(m_control_transmit_buffer->lock());
  170. PhysicalAddress start_of_chunk;
  171. size_t length_of_chunk;
  172. auto data = UserOrKernelBuffer::for_kernel_buffer((u8*)&message);
  173. while (!m_control_transmit_buffer->copy_data_in(data, 0, sizeof(message), start_of_chunk, length_of_chunk)) {
  174. ringbuffer_lock.unlock();
  175. m_control_wait_queue.wait_forever();
  176. ringbuffer_lock.lock();
  177. }
  178. auto& queue = get_queue(CONTROL_TRANSMITQ);
  179. SpinlockLocker queue_lock(queue.lock());
  180. QueueChain chain(queue);
  181. bool did_add_buffer = chain.add_buffer_to_chain(start_of_chunk, length_of_chunk, BufferType::DeviceReadable);
  182. VERIFY(did_add_buffer);
  183. supply_chain_and_notify(CONTROL_TRANSMITQ, chain);
  184. }
  185. void Console::send_open_control_message(unsigned port_number, bool open)
  186. {
  187. ControlMessage port_open {
  188. .id = static_cast<u32>(port_number),
  189. .event = (u16)ControlEvent::PortOpen,
  190. .value = open
  191. };
  192. write_control_message(port_open);
  193. }
  194. }