/* * Copyright 2008-2011 Michael Lotz * Distributed under the terms of the MIT license. */ //! Driver for USB Human Interface Devices. #include "Driver.h" #include "MouseProtocolHandler.h" #include "HIDCollection.h" #include "HIDDevice.h" #include "HIDReport.h" #include "HIDReportItem.h" #include #include #include #include #include MouseProtocolHandler::MouseProtocolHandler(HIDReport &report, HIDReportItem &xAxis, HIDReportItem &yAxis) : ProtocolHandler(report.Device(), "input/mouse/" DEVICE_PATH_SUFFIX "/", 0), fReport(report), fXAxis(xAxis), fYAxis(yAxis), fWheel(NULL), fHorizontalPan(NULL), fLastButtons(0), fClickCount(0), fLastClickTime(0), fClickSpeed(250000) { uint32 buttonCount = 0; for (uint32 i = 0; i < report.CountItems(); i++) { HIDReportItem *item = report.ItemAt(i); if (!item->HasData()) continue; if (item->UsagePage() == B_HID_USAGE_PAGE_BUTTON && item->UsageID() - 1 < B_MAX_MOUSE_BUTTONS) { fButtons[buttonCount++] = item; } } fButtons[buttonCount] = NULL; fWheel = report.FindItem(B_HID_USAGE_PAGE_GENERIC_DESKTOP, B_HID_UID_GD_WHEEL); fHorizontalPan = report.FindItem(B_HID_USAGE_PAGE_CONSUMER, B_HID_UID_CON_AC_PAN); TRACE("mouse device with %" B_PRIu32 " buttons %sand %swheel\n", buttonCount, fHorizontalPan == NULL ? "" : ", horizontal pan ", fWheel == NULL ? "no " : ""); TRACE("report id: %u\n", report.ID()); } void MouseProtocolHandler::AddHandlers(HIDDevice &device, HIDCollection &collection, ProtocolHandler *&handlerList) { bool supported = false; switch (collection.UsagePage()) { case B_HID_USAGE_PAGE_GENERIC_DESKTOP: { switch (collection.UsageID()) { case B_HID_UID_GD_MOUSE: case B_HID_UID_GD_POINTER: supported = true; break; } break; } } if (!supported) { TRACE("collection not a mouse\n"); return; } HIDParser &parser = device.Parser(); uint32 maxReportCount = parser.CountReports(HID_REPORT_TYPE_INPUT); if (maxReportCount == 0) return; uint32 inputReportCount = 0; HIDReport *inputReports[maxReportCount]; collection.BuildReportList(HID_REPORT_TYPE_INPUT, inputReports, inputReportCount); for (uint32 i = 0; i < inputReportCount; i++) { HIDReport *inputReport = inputReports[i]; // try to find at least an absolute x and y axis HIDReportItem *xAxis = inputReport->FindItem( B_HID_USAGE_PAGE_GENERIC_DESKTOP, B_HID_UID_GD_X); if (xAxis == NULL || !xAxis->Relative()) continue; HIDReportItem *yAxis = inputReport->FindItem( B_HID_USAGE_PAGE_GENERIC_DESKTOP, B_HID_UID_GD_Y); if (yAxis == NULL || !yAxis->Relative()) continue; ProtocolHandler *newHandler = new(std::nothrow) MouseProtocolHandler( *inputReport, *xAxis, *yAxis); if (newHandler == NULL) { TRACE("failed to allocated mouse protocol handler\n"); continue; } newHandler->SetNextHandler(handlerList); handlerList = newHandler; } } status_t MouseProtocolHandler::Control(uint32 *cookie, uint32 op, void *buffer, size_t length) { switch (op) { case B_GET_DEVICE_NAME: { const char name[] = DEVICE_NAME" Mouse"; return IOGetDeviceName(name,buffer,length); } case MS_READ: { if (length < sizeof(mouse_movement)) return B_BUFFER_OVERFLOW; while (true) { mouse_movement movement; status_t result = _ReadReport(&movement, cookie); if (result == B_INTERRUPTED) continue; if (!IS_USER_ADDRESS(buffer) || user_memcpy(buffer, &movement, sizeof(movement)) != B_OK) { return B_BAD_ADDRESS; } return result; } } case MS_NUM_EVENTS: { if (fReport.Device()->IsRemoved()) return B_DEV_NOT_READY; // we are always on demand, so 0 queued events return 0; } case MS_SET_CLICKSPEED: if (!IS_USER_ADDRESS(buffer) || user_memcpy(&fClickSpeed, buffer, sizeof(bigtime_t)) != B_OK) { return B_BAD_ADDRESS; } return B_OK; } return B_ERROR; } status_t MouseProtocolHandler::_ReadReport(void *buffer, uint32 *cookie) { status_t result = fReport.WaitForReport(B_INFINITE_TIMEOUT); if (result != B_OK) { if (fReport.Device()->IsRemoved()) { TRACE("device has been removed\n"); return B_DEV_NOT_READY; } if ((*cookie & PROTOCOL_HANDLER_COOKIE_FLAG_CLOSED) != 0) return B_CANCELED; if (result != B_INTERRUPTED) { // interrupts happen when other reports come in on the same // input as ours TRACE_ALWAYS("error waiting for report: %s\n", strerror(result)); } // signal that we simply want to try again return B_INTERRUPTED; } uint32 axisRelativeData[2]; axisRelativeData[0] = 0; axisRelativeData[1] = 0; if (fXAxis.Extract() == B_OK && fXAxis.Valid()) axisRelativeData[0] = fXAxis.Data(); if (fYAxis.Extract() == B_OK && fYAxis.Valid()) axisRelativeData[1] = fYAxis.Data(); uint32 wheelData[2] = {0}; if (fWheel != NULL && fWheel->Extract() == B_OK && fWheel->Valid()) wheelData[0] = fWheel->Data(); if (fHorizontalPan != NULL && fHorizontalPan->Extract() == B_OK && fHorizontalPan->Valid()) wheelData[1] = fHorizontalPan->Data(); uint32 buttons = 0; for (uint32 i = 0; i < B_MAX_MOUSE_BUTTONS; i++) { HIDReportItem *button = fButtons[i]; if (button == NULL) break; if (button->Extract() == B_OK && button->Valid()) buttons |= (button->Data() & 1) << (button->UsageID() - 1); } fReport.DoneProcessing(); TRACE("got mouse report\n"); int32 clicks = 0; bigtime_t timestamp = system_time(); if (buttons != 0) { if (fLastButtons == 0) { if (fLastClickTime + fClickSpeed > timestamp) fClickCount++; else fClickCount = 1; } fLastClickTime = timestamp; clicks = fClickCount; } fLastButtons = buttons; mouse_movement *info = (mouse_movement *)buffer; memset(info, 0, sizeof(mouse_movement)); info->buttons = buttons; info->xdelta = axisRelativeData[0]; info->ydelta = -axisRelativeData[1]; info->clicks = clicks; info->timestamp = timestamp; info->wheel_ydelta = -wheelData[0]; info->wheel_xdelta = wheelData[1]; return B_OK; }