USBPipe.cpp 2.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384
  1. /*
  2. * Copyright (c) 2021, Jesse Buhagiar <jooster669@gmail.com>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <Kernel/Bus/USB/PacketTypes.h>
  7. #include <Kernel/Bus/USB/UHCIController.h>
  8. #include <Kernel/Bus/USB/USBPipe.h>
  9. #include <Kernel/Bus/USB/USBTransfer.h>
  10. namespace Kernel::USB {
  11. KResultOr<NonnullOwnPtr<Pipe>> Pipe::try_create_pipe(Type type, Direction direction, u8 endpoint_address, u16 max_packet_size, i8 device_address, u8 poll_interval)
  12. {
  13. auto pipe = adopt_own_if_nonnull(new (nothrow) Pipe(type, direction, endpoint_address, max_packet_size, device_address, poll_interval));
  14. if (!pipe)
  15. return ENOMEM;
  16. return pipe.release_nonnull();
  17. }
  18. Pipe::Pipe(Type type, Pipe::Direction direction, u16 max_packet_size)
  19. : m_type(type)
  20. , m_direction(direction)
  21. , m_endpoint_address(0)
  22. , m_max_packet_size(max_packet_size)
  23. , m_poll_interval(0)
  24. , m_data_toggle(false)
  25. {
  26. }
  27. Pipe::Pipe(Type type, Direction direction, USBEndpointDescriptor& endpoint [[maybe_unused]])
  28. : m_type(type)
  29. , m_direction(direction)
  30. {
  31. // TODO: decode endpoint structure
  32. }
  33. Pipe::Pipe(Type type, Direction direction, u8 endpoint_address, u16 max_packet_size, u8 poll_interval, i8 device_address)
  34. : m_type(type)
  35. , m_direction(direction)
  36. , m_device_address(device_address)
  37. , m_endpoint_address(endpoint_address)
  38. , m_max_packet_size(max_packet_size)
  39. , m_poll_interval(poll_interval)
  40. , m_data_toggle(false)
  41. {
  42. }
  43. KResultOr<size_t> Pipe::control_transfer(u8 request_type, u8 request, u16 value, u16 index, u16 length, void* data)
  44. {
  45. USBRequestData usb_request;
  46. usb_request.request_type = request_type;
  47. usb_request.request = request;
  48. usb_request.value = value;
  49. usb_request.index = index;
  50. usb_request.length = length;
  51. auto transfer = Transfer::try_create(*this, length);
  52. if (!transfer)
  53. return ENOMEM;
  54. transfer->set_setup_packet(usb_request);
  55. dbgln_if(USB_DEBUG, "Pipe: Transfer allocated @ {:08x}", transfer->buffer_physical());
  56. auto transfer_len_or_error = UHCIController::the().submit_control_transfer(*transfer);
  57. if (transfer_len_or_error.is_error())
  58. return transfer_len_or_error.error();
  59. auto transfer_length = transfer_len_or_error.release_value();
  60. // TODO: Check transfer for completion and copy data from transfer buffer into data
  61. if (length > 0)
  62. memcpy(reinterpret_cast<u8*>(data), transfer->buffer().as_ptr() + sizeof(USBRequestData), length);
  63. dbgln_if(USB_DEBUG, "Pipe: Control Transfer complete!");
  64. return transfer_length;
  65. }
  66. }