* Implemented transparent device replugs. If the same device (by MAC address)

is plugged in after having been unplugged the device will now reuse a still
  existing ECMDevice object. This allows for the link simply going down when a
  device is unplugged and going up again when the device is replugged. The
  nice thing is that due to the way our usb drivers work it doesn't matter
  where you replug the device, so you can switch it from one port to another
  or even from a highspeed to a fullspeed bus transparently.
* Fix a race condition between the notify hook and the device removed hook.
* Added some comments and extended some debug output to be more useful.

git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@25464 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
Michael Lotz 2008-05-12 11:22:36 +00:00
parent c1d860fb1f
commit f97c4f2fdc
3 changed files with 234 additions and 122 deletions

View File

@ -28,14 +28,30 @@ usb_ecm_device_added(usb_device device, void **cookie)
{ {
*cookie = NULL; *cookie = NULL;
// check if this is a replug of an existing device first
mutex_lock(&gDriverLock);
for (int32 i = 0; i < MAX_DEVICES; i++) {
if (gECMDevices[i] == NULL)
continue;
if (gECMDevices[i]->CompareAndReattach(device) != B_OK)
continue;
TRACE_ALWAYS("ecm device %ld replugged\n", i);
*cookie = gECMDevices[i];
mutex_unlock(&gDriverLock);
return B_OK;
}
// no such device yet, create a new one
ECMDevice *ecmDevice = new ECMDevice(device); ECMDevice *ecmDevice = new ECMDevice(device);
status_t status = ecmDevice->InitCheck(); status_t status = ecmDevice->InitCheck();
if (status < B_OK) { if (status < B_OK) {
delete ecmDevice; delete ecmDevice;
mutex_unlock(&gDriverLock);
return status; return status;
} }
mutex_lock(&gDriverLock);
for (int32 i = 0; i < MAX_DEVICES; i++) { for (int32 i = 0; i < MAX_DEVICES; i++) {
if (gECMDevices[i] != NULL) if (gECMDevices[i] != NULL)
continue; continue;
@ -156,7 +172,7 @@ usb_ecm_open(const char *name, uint32 flags, void **cookie)
status_t status = ENODEV; status_t status = ENODEV;
int32 index = strtol(name + strlen(sDeviceBaseName), NULL, 10); int32 index = strtol(name + strlen(sDeviceBaseName), NULL, 10);
if (index >= 0 && index < MAX_DEVICES && gECMDevices[index]) { if (index >= 0 && index < MAX_DEVICES && gECMDevices[index]) {
status = gECMDevices[index]->Open(flags); status = gECMDevices[index]->Open();
*cookie = gECMDevices[index]; *cookie = gECMDevices[index];
} }

View File

@ -16,6 +16,7 @@ ECMDevice::ECMDevice(usb_device device)
: fStatus(B_ERROR), : fStatus(B_ERROR),
fOpen(false), fOpen(false),
fRemoved(false), fRemoved(false),
fInsideNotify(0),
fDevice(device), fDevice(device),
fControlInterfaceIndex(0), fControlInterfaceIndex(0),
fDataInterfaceIndex(0), fDataInterfaceIndex(0),
@ -35,87 +36,15 @@ ECMDevice::ECMDevice(usb_device device)
{ {
const usb_device_descriptor *deviceDescriptor const usb_device_descriptor *deviceDescriptor
= gUSBModule->get_device_descriptor(device); = gUSBModule->get_device_descriptor(device);
const usb_configuration_info *config
= gUSBModule->get_nth_configuration(device, 0);
if (deviceDescriptor == NULL || config == NULL) { if (deviceDescriptor == NULL) {
TRACE_ALWAYS("failed to get basic device info\n"); TRACE_ALWAYS("failed to get device descriptor\n");
return; return;
} }
TRACE_ALWAYS("creating device: vendor: 0x%04x; device: 0x%04x\n", fVendorID = deviceDescriptor->vendor_id;
deviceDescriptor->vendor_id, deviceDescriptor->product_id); fProductID = deviceDescriptor->product_id;
uint8 controlIndex = 0;
uint8 dataIndex = 0;
bool foundUnionDescriptor = false;
bool foundEthernetDescriptor = false;
for (size_t i = 0; i < config->interface_count
&& (!foundUnionDescriptor || !foundEthernetDescriptor); i++) {
usb_interface_info *interface = config->interface[i].active;
usb_interface_descriptor *descriptor = interface->descr;
if (descriptor->interface_class == USB_INTERFACE_CLASS_CDC
&& descriptor->interface_subclass == USB_INTERFACE_SUBCLASS_ECM
&& interface->generic_count > 0) {
// try to find and interpret the union and ethernet functional
// descriptors
foundUnionDescriptor = foundEthernetDescriptor = false;
for (size_t j = 0; j < interface->generic_count; j++) {
usb_generic_descriptor *generic = &interface->generic[j]->generic;
if (generic->length >= 5
&& generic->data[0] == FUNCTIONAL_SUBTYPE_UNION) {
controlIndex = generic->data[1];
dataIndex = generic->data[2];
foundUnionDescriptor = true;
} else if (generic->length >= sizeof(ethernet_functional_descriptor)
&& generic->data[0] == FUNCTIONAL_SUBTYPE_ETHERNET) {
ethernet_functional_descriptor *ethernet
= (ethernet_functional_descriptor *)generic->data;
fMACAddressIndex = ethernet->mac_address_index;
fMaxSegmentSize = ethernet->max_segment_size;
foundEthernetDescriptor = true;
}
if (foundUnionDescriptor && foundEthernetDescriptor)
break;
}
}
}
if (!foundUnionDescriptor) {
TRACE_ALWAYS("did not find a union descriptor\n");
return;
}
if (!foundEthernetDescriptor) {
TRACE_ALWAYS("did not find an ethernet descriptor\n");
return;
}
if (_ReadMACAddress() != B_OK) {
TRACE_ALWAYS("failed to read mac address\n");
return;
}
if (controlIndex >= config->interface_count) {
TRACE_ALWAYS("control interface index invalid\n");
return;
}
// check that the indicated control interface fits our needs
usb_interface_info *interface = config->interface[controlIndex].active;
usb_interface_descriptor *descriptor = interface->descr;
if ((descriptor->interface_class != USB_INTERFACE_CLASS_CDC
|| descriptor->interface_subclass != USB_INTERFACE_SUBCLASS_ECM)
|| interface->endpoint_count == 0) {
TRACE_ALWAYS("control interface invalid\n");
return;
}
fControlInterfaceIndex = controlIndex;
fNotifyEndpoint = interface->endpoint[0].handle;
// setup notify buffer and try to schedule a notification transfer
fNotifyBufferLength = 64; fNotifyBufferLength = 64;
fNotifyBuffer = (uint8 *)malloc(fNotifyBufferLength); fNotifyBuffer = (uint8 *)malloc(fNotifyBufferLength);
if (fNotifyBuffer == NULL) { if (fNotifyBuffer == NULL) {
@ -123,35 +52,6 @@ ECMDevice::ECMDevice(usb_device device)
return; return;
} }
if (gUSBModule->queue_interrupt(fNotifyEndpoint, fNotifyBuffer,
fNotifyBufferLength, _NotifyCallback, this) != B_OK) {
// we cannot use notifications - hardcode to active connection
fHasConnection = true;
fDownstreamSpeed = 1000 * 1000 * 10; // 10Mbps
fUpstreamSpeed = 1000 * 1000 * 10; // 10Mbps
}
if (dataIndex >= config->interface_count) {
TRACE_ALWAYS("data interface index invalid\n");
return;
}
// check that the indicated data interface fits our needs
if (config->interface[dataIndex].alt_count < 2) {
TRACE_ALWAYS("data interface does not provide two alternate interfaces\n");
return;
}
// alternate 0 is the disabled, endpoint-less default interface
interface = &config->interface[dataIndex].alt[1];
descriptor = interface->descr;
if (descriptor->interface_class != USB_INTERFACE_CLASS_CDC_DATA
|| interface->endpoint_count < 2) {
TRACE_ALWAYS("data interface invalid\n");
return;
}
fDataInterfaceIndex = dataIndex;
fNotifyReadSem = create_sem(0, DRIVER_NAME"_notify_read"); fNotifyReadSem = create_sem(0, DRIVER_NAME"_notify_read");
if (fNotifyReadSem < B_OK) { if (fNotifyReadSem < B_OK) {
TRACE_ALWAYS("failed to create read notify sem\n"); TRACE_ALWAYS("failed to create read notify sem\n");
@ -164,6 +64,16 @@ ECMDevice::ECMDevice(usb_device device)
return; return;
} }
if (_SetupDevice() != B_OK) {
TRACE_ALWAYS("failed to setup device\n");
return;
}
if (_ReadMACAddress(fDevice, fMACAddress) != B_OK) {
TRACE_ALWAYS("failed to read mac address\n");
return;
}
fStatus = B_OK; fStatus = B_OK;
} }
@ -175,16 +85,20 @@ ECMDevice::~ECMDevice()
if (fNotifyWriteSem >= B_OK) if (fNotifyWriteSem >= B_OK)
delete_sem(fNotifyWriteSem); delete_sem(fNotifyWriteSem);
gUSBModule->cancel_queued_transfers(fNotifyEndpoint); if (!fRemoved)
gUSBModule->cancel_queued_transfers(fNotifyEndpoint);
free(fNotifyBuffer); free(fNotifyBuffer);
} }
status_t status_t
ECMDevice::Open(uint32 flags) ECMDevice::Open()
{ {
if (fOpen) if (fOpen)
return B_BUSY; return B_BUSY;
if (fRemoved)
return B_ERROR;
// reset the device by switching the data interface to the disabled first // reset the device by switching the data interface to the disabled first
// interface and then enable it by setting the second actual data interface // interface and then enable it by setting the second actual data interface
@ -280,12 +194,12 @@ ECMDevice::Read(uint8 *buffer, size_t *numBytes)
return result; return result;
} }
if (fStatusRead != B_OK && fStatusRead != B_CANCELED) { if (fStatusRead != B_OK && fStatusRead != B_CANCELED && !fRemoved) {
TRACE_ALWAYS("device status error 0x%08lx\n", fStatusRead); TRACE_ALWAYS("device status error 0x%08lx\n", fStatusRead);
result = gUSBModule->clear_feature(fReadEndpoint, result = gUSBModule->clear_feature(fReadEndpoint,
USB_FEATURE_ENDPOINT_HALT); USB_FEATURE_ENDPOINT_HALT);
if (result != B_OK) { if (result != B_OK) {
TRACE_ALWAYS("failed to clear halt state\n"); TRACE_ALWAYS("failed to clear halt state on read\n");
*numBytes = 0; *numBytes = 0;
return result; return result;
} }
@ -317,12 +231,12 @@ ECMDevice::Write(const uint8 *buffer, size_t *numBytes)
return result; return result;
} }
if (fStatusWrite != B_OK && fStatusWrite != B_CANCELED) { if (fStatusWrite != B_OK && fStatusWrite != B_CANCELED && !fRemoved) {
TRACE_ALWAYS("device status error 0x%08lx\n", fStatusWrite); TRACE_ALWAYS("device status error 0x%08lx\n", fStatusWrite);
result = gUSBModule->clear_feature(fWriteEndpoint, result = gUSBModule->clear_feature(fWriteEndpoint,
USB_FEATURE_ENDPOINT_HALT); USB_FEATURE_ENDPOINT_HALT);
if (result != B_OK) { if (result != B_OK) {
TRACE_ALWAYS("failed to clear halt state\n"); TRACE_ALWAYS("failed to clear halt state on write\n");
*numBytes = 0; *numBytes = 0;
return result; return result;
} }
@ -379,13 +293,186 @@ ECMDevice::Removed()
fHasConnection = false; fHasConnection = false;
fDownstreamSpeed = fUpstreamSpeed = 0; fDownstreamSpeed = fUpstreamSpeed = 0;
// the notify hook is different from the read and write hooks as it does
// itself schedule traffic (while the other hooks only release a semaphore
// to notify another thread which in turn safly checks for the removed
// case) - so we must ensure that we are not inside the notify hook anymore
// before returning, as we would otherwise violate the promise not to use
// any of the pipes after returning from the removed hook
while (atomic_add(&fInsideNotify, 0) != 0)
snooze(100);
gUSBModule->cancel_queued_transfers(fNotifyEndpoint);
gUSBModule->cancel_queued_transfers(fReadEndpoint);
gUSBModule->cancel_queued_transfers(fWriteEndpoint);
if (fLinkStateChangeSem >= B_OK) if (fLinkStateChangeSem >= B_OK)
release_sem_etc(fLinkStateChangeSem, 1, B_DO_NOT_RESCHEDULE); release_sem_etc(fLinkStateChangeSem, 1, B_DO_NOT_RESCHEDULE);
} }
status_t status_t
ECMDevice::_ReadMACAddress() ECMDevice::CompareAndReattach(usb_device device)
{
const usb_device_descriptor *deviceDescriptor
= gUSBModule->get_device_descriptor(device);
if (deviceDescriptor == NULL) {
TRACE_ALWAYS("failed to get device descriptor\n");
return B_ERROR;
}
if (deviceDescriptor->vendor_id != fVendorID
&& deviceDescriptor->product_id != fProductID) {
// this certainly isn't the same device
return B_BAD_VALUE;
}
// this might be the same device that was replugged - read the MAC address
// (which should be at the same index) to make sure
uint8 macBuffer[6];
if (_ReadMACAddress(device, macBuffer) != B_OK
|| memcmp(macBuffer, fMACAddress, sizeof(macBuffer)) != 0) {
// reading the MAC address failed or they are not the same
return B_BAD_VALUE;
}
// this is the same device that was replugged - clear the removed state,
// re-setup the endpoints and transfers and open the device if it was
// previously opened
fDevice = device;
fRemoved = false;
status_t result = _SetupDevice();
if (result != B_OK) {
fRemoved = true;
return result;
}
// in case notifications do not work we will have a hardcoded connection
// need to register that and notify the network stack ourselfs if this is
// the case as the open will not result in a corresponding notification
bool noNotifications = fHasConnection;
if (fOpen) {
fOpen = false;
result = Open();
if (result == B_OK && noNotifications && fLinkStateChangeSem >= B_OK)
release_sem_etc(fLinkStateChangeSem, 1, B_DO_NOT_RESCHEDULE);
}
return B_OK;
}
status_t
ECMDevice::_SetupDevice()
{
const usb_configuration_info *config
= gUSBModule->get_nth_configuration(fDevice, 0);
if (config == NULL) {
TRACE_ALWAYS("failed to get device configuration\n");
return B_ERROR;
}
uint8 controlIndex = 0;
uint8 dataIndex = 0;
bool foundUnionDescriptor = false;
bool foundEthernetDescriptor = false;
for (size_t i = 0; i < config->interface_count
&& (!foundUnionDescriptor || !foundEthernetDescriptor); i++) {
usb_interface_info *interface = config->interface[i].active;
usb_interface_descriptor *descriptor = interface->descr;
if (descriptor->interface_class == USB_INTERFACE_CLASS_CDC
&& descriptor->interface_subclass == USB_INTERFACE_SUBCLASS_ECM
&& interface->generic_count > 0) {
// try to find and interpret the union and ethernet functional
// descriptors
foundUnionDescriptor = foundEthernetDescriptor = false;
for (size_t j = 0; j < interface->generic_count; j++) {
usb_generic_descriptor *generic = &interface->generic[j]->generic;
if (generic->length >= 5
&& generic->data[0] == FUNCTIONAL_SUBTYPE_UNION) {
controlIndex = generic->data[1];
dataIndex = generic->data[2];
foundUnionDescriptor = true;
} else if (generic->length >= sizeof(ethernet_functional_descriptor)
&& generic->data[0] == FUNCTIONAL_SUBTYPE_ETHERNET) {
ethernet_functional_descriptor *ethernet
= (ethernet_functional_descriptor *)generic->data;
fMACAddressIndex = ethernet->mac_address_index;
fMaxSegmentSize = ethernet->max_segment_size;
foundEthernetDescriptor = true;
}
if (foundUnionDescriptor && foundEthernetDescriptor)
break;
}
}
}
if (!foundUnionDescriptor) {
TRACE_ALWAYS("did not find a union descriptor\n");
return B_ERROR;
}
if (!foundEthernetDescriptor) {
TRACE_ALWAYS("did not find an ethernet descriptor\n");
return B_ERROR;
}
if (controlIndex >= config->interface_count) {
TRACE_ALWAYS("control interface index invalid\n");
return B_ERROR;
}
// check that the indicated control interface fits our needs
usb_interface_info *interface = config->interface[controlIndex].active;
usb_interface_descriptor *descriptor = interface->descr;
if ((descriptor->interface_class != USB_INTERFACE_CLASS_CDC
|| descriptor->interface_subclass != USB_INTERFACE_SUBCLASS_ECM)
|| interface->endpoint_count == 0) {
TRACE_ALWAYS("control interface invalid\n");
return B_ERROR;
}
fControlInterfaceIndex = controlIndex;
fNotifyEndpoint = interface->endpoint[0].handle;
if (gUSBModule->queue_interrupt(fNotifyEndpoint, fNotifyBuffer,
fNotifyBufferLength, _NotifyCallback, this) != B_OK) {
// we cannot use notifications - hardcode to active connection
fHasConnection = true;
fDownstreamSpeed = 1000 * 1000 * 10; // 10Mbps
fUpstreamSpeed = 1000 * 1000 * 10; // 10Mbps
}
if (dataIndex >= config->interface_count) {
TRACE_ALWAYS("data interface index invalid\n");
return B_ERROR;
}
// check that the indicated data interface fits our needs
if (config->interface[dataIndex].alt_count < 2) {
TRACE_ALWAYS("data interface does not provide two alternate interfaces\n");
return B_ERROR;
}
// alternate 0 is the disabled, endpoint-less default interface
interface = &config->interface[dataIndex].alt[1];
descriptor = interface->descr;
if (descriptor->interface_class != USB_INTERFACE_CLASS_CDC_DATA
|| interface->endpoint_count < 2) {
TRACE_ALWAYS("data interface invalid\n");
return B_ERROR;
}
fDataInterfaceIndex = dataIndex;
return B_OK;
}
status_t
ECMDevice::_ReadMACAddress(usb_device device, uint8 *buffer)
{ {
if (fMACAddressIndex == 0) if (fMACAddressIndex == 0)
return B_BAD_VALUE; return B_BAD_VALUE;
@ -393,7 +480,7 @@ ECMDevice::_ReadMACAddress()
size_t actualLength = 0; size_t actualLength = 0;
size_t macStringLength = 26; size_t macStringLength = 26;
uint8 macString[macStringLength]; uint8 macString[macStringLength];
status_t result = gUSBModule->get_descriptor(fDevice, USB_DESCRIPTOR_STRING, status_t result = gUSBModule->get_descriptor(device, USB_DESCRIPTOR_STRING,
fMACAddressIndex, 0, macString, macStringLength, &actualLength); fMACAddressIndex, 0, macString, macStringLength, &actualLength);
if (result != B_OK) if (result != B_OK)
return result; return result;
@ -408,12 +495,11 @@ ECMDevice::_ReadMACAddress()
for (int32 i = 0; i < 6; i++) { for (int32 i = 0; i < 6; i++) {
macPart[0] = macString[2 + i * 4 + 0]; macPart[0] = macString[2 + i * 4 + 0];
macPart[1] = macString[2 + i * 4 + 2]; macPart[1] = macString[2 + i * 4 + 2];
fMACAddress[i] = strtol(macPart, NULL, 16); buffer[i] = strtol(macPart, NULL, 16);
} }
TRACE_ALWAYS("read mac address: %02x:%02x:%02x:%02x:%02x:%02x\n", TRACE_ALWAYS("read mac address: %02x:%02x:%02x:%02x:%02x:%02x\n",
fMACAddress[0], fMACAddress[1], fMACAddress[2], fMACAddress[3], buffer[0], buffer[1], buffer[2], buffer[3], buffer[4], buffer[5]);
fMACAddress[4], fMACAddress[5]);
return B_OK; return B_OK;
} }
@ -445,8 +531,11 @@ ECMDevice::_NotifyCallback(void *cookie, int32 status, void *data,
uint32 actualLength) uint32 actualLength)
{ {
ECMDevice *device = (ECMDevice *)cookie; ECMDevice *device = (ECMDevice *)cookie;
if (status == B_CANCELED || device->fRemoved) atomic_add(&device->fInsideNotify, 1);
if (status == B_CANCELED || device->fRemoved) {
atomic_add(&device->fInsideNotify, -1);
return; return;
}
if (status == B_OK && actualLength >= sizeof(cdc_notification)) { if (status == B_OK && actualLength >= sizeof(cdc_notification)) {
bool linkStateChange = false; bool linkStateChange = false;
@ -494,10 +583,11 @@ ECMDevice::_NotifyCallback(void *cookie, int32 status, void *data,
TRACE_ALWAYS("device status error 0x%08lx\n", status); TRACE_ALWAYS("device status error 0x%08lx\n", status);
if (gUSBModule->clear_feature(device->fNotifyEndpoint, if (gUSBModule->clear_feature(device->fNotifyEndpoint,
USB_FEATURE_ENDPOINT_HALT) != B_OK) USB_FEATURE_ENDPOINT_HALT) != B_OK)
TRACE_ALWAYS("failed to clear halt state\n"); TRACE_ALWAYS("failed to clear halt state in notify hook\n");
} }
// schedule next notification buffer // schedule next notification buffer
gUSBModule->queue_interrupt(device->fNotifyEndpoint, device->fNotifyBuffer, gUSBModule->queue_interrupt(device->fNotifyEndpoint, device->fNotifyBuffer,
device->fNotifyBufferLength, _NotifyCallback, device); device->fNotifyBufferLength, _NotifyCallback, device);
atomic_add(&device->fInsideNotify, -1);
} }

View File

@ -15,7 +15,7 @@ public:
status_t InitCheck() { return fStatus; }; status_t InitCheck() { return fStatus; };
status_t Open(uint32 flags); status_t Open();
bool IsOpen() { return fOpen; }; bool IsOpen() { return fOpen; };
status_t Close(); status_t Close();
@ -28,6 +28,8 @@ public:
void Removed(); void Removed();
bool IsRemoved() { return fRemoved; }; bool IsRemoved() { return fRemoved; };
status_t CompareAndReattach(usb_device device);
private: private:
static void _ReadCallback(void *cookie, int32 status, static void _ReadCallback(void *cookie, int32 status,
void *data, uint32 actualLength); void *data, uint32 actualLength);
@ -36,13 +38,17 @@ static void _WriteCallback(void *cookie, int32 status,
static void _NotifyCallback(void *cookie, int32 status, static void _NotifyCallback(void *cookie, int32 status,
void *data, uint32 actualLength); void *data, uint32 actualLength);
status_t _ReadMACAddress(); status_t _SetupDevice();
status_t _ReadMACAddress(usb_device device, uint8 *buffer);
// state tracking // state tracking
status_t fStatus; status_t fStatus;
bool fOpen; bool fOpen;
bool fRemoved; bool fRemoved;
vint32 fInsideNotify;
usb_device fDevice; usb_device fDevice;
uint16 fVendorID;
uint16 fProductID;
// interface and device infos // interface and device infos
uint8 fControlInterfaceIndex; uint8 fControlInterfaceIndex;