Support hot unplugging
This commit is contained in:
parent
60c1ffdfdc
commit
52d8255d20
7 changed files with 120 additions and 51 deletions
|
@ -22,7 +22,7 @@
|
||||||
#include <grub/usb.h>
|
#include <grub/usb.h>
|
||||||
#include <grub/misc.h>
|
#include <grub/misc.h>
|
||||||
#include <grub/list.h>
|
#include <grub/list.h>
|
||||||
#include <grub/dl.h>
|
#include <grub/term.h>
|
||||||
|
|
||||||
static grub_usb_controller_dev_t grub_usb_list;
|
static grub_usb_controller_dev_t grub_usb_list;
|
||||||
struct grub_usb_attach_desc *attach_hooks;
|
struct grub_usb_attach_desc *attach_hooks;
|
||||||
|
@ -334,3 +334,14 @@ grub_usb_unregister_attach_hook_class (struct grub_usb_attach_desc *desc)
|
||||||
{
|
{
|
||||||
grub_list_remove (GRUB_AS_LIST_P (&attach_hooks), GRUB_AS_LIST (desc));
|
grub_list_remove (GRUB_AS_LIST_P (&attach_hooks), GRUB_AS_LIST (desc));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
GRUB_MOD_INIT(usb)
|
||||||
|
{
|
||||||
|
grub_term_poll_usb = grub_usb_poll_devices;
|
||||||
|
}
|
||||||
|
|
||||||
|
GRUB_MOD_FINI(usb)
|
||||||
|
{
|
||||||
|
grub_term_poll_usb = NULL;
|
||||||
|
}
|
||||||
|
|
101
bus/usb/usbhub.c
101
bus/usb/usbhub.c
|
@ -33,7 +33,7 @@ struct grub_usb_hub
|
||||||
struct grub_usb_hub *next;
|
struct grub_usb_hub *next;
|
||||||
grub_usb_controller_t controller;
|
grub_usb_controller_t controller;
|
||||||
int nports;
|
int nports;
|
||||||
grub_usb_speed_t *speed;
|
struct grub_usb_device **devices;
|
||||||
grub_usb_device_t dev;
|
grub_usb_device_t dev;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -104,7 +104,7 @@ grub_usb_hub_add_dev (grub_usb_controller_t controller, grub_usb_speed_t speed)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static grub_err_t
|
static grub_usb_err_t
|
||||||
grub_usb_add_hub (grub_usb_device_t dev)
|
grub_usb_add_hub (grub_usb_device_t dev)
|
||||||
{
|
{
|
||||||
struct grub_usb_usb_hubdesc hubdesc;
|
struct grub_usb_usb_hubdesc hubdesc;
|
||||||
|
@ -112,6 +112,7 @@ grub_usb_add_hub (grub_usb_device_t dev)
|
||||||
int i;
|
int i;
|
||||||
grub_uint64_t timeout;
|
grub_uint64_t timeout;
|
||||||
grub_usb_device_t next_dev;
|
grub_usb_device_t next_dev;
|
||||||
|
grub_usb_device_t *attached_devices;
|
||||||
|
|
||||||
err = grub_usb_control_msg (dev, (GRUB_USB_REQTYPE_IN
|
err = grub_usb_control_msg (dev, (GRUB_USB_REQTYPE_IN
|
||||||
| GRUB_USB_REQTYPE_CLASS
|
| GRUB_USB_REQTYPE_CLASS
|
||||||
|
@ -130,6 +131,13 @@ grub_usb_add_hub (grub_usb_device_t dev)
|
||||||
grub_dprintf ("usb", "Hub set configuration\n");
|
grub_dprintf ("usb", "Hub set configuration\n");
|
||||||
grub_usb_set_configuration (dev, 1);
|
grub_usb_set_configuration (dev, 1);
|
||||||
|
|
||||||
|
attached_devices = grub_zalloc (hubdesc.portcnt
|
||||||
|
* sizeof (attached_devices[0]));
|
||||||
|
if (!attached_devices)
|
||||||
|
return GRUB_USB_ERR_INTERNAL;
|
||||||
|
dev->children = attached_devices;
|
||||||
|
dev->nports = hubdesc.portcnt;
|
||||||
|
|
||||||
/* Power on all Hub ports. */
|
/* Power on all Hub ports. */
|
||||||
for (i = 1; i <= hubdesc.portcnt; i++)
|
for (i = 1; i <= hubdesc.portcnt; i++)
|
||||||
{
|
{
|
||||||
|
@ -236,6 +244,8 @@ grub_usb_add_hub (grub_usb_device_t dev)
|
||||||
if (! next_dev)
|
if (! next_dev)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
|
attached_devices[i - 1] = next_dev;
|
||||||
|
|
||||||
/* If the device is a Hub, scan it for more devices. */
|
/* If the device is a Hub, scan it for more devices. */
|
||||||
if (next_dev->descdev.class == 0x09)
|
if (next_dev->descdev.class == 0x09)
|
||||||
grub_usb_add_hub (next_dev);
|
grub_usb_add_hub (next_dev);
|
||||||
|
@ -246,27 +256,29 @@ grub_usb_add_hub (grub_usb_device_t dev)
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
attach_root_port (grub_usb_controller_t controller, int portno,
|
attach_root_port (struct grub_usb_hub *hub, int portno,
|
||||||
grub_usb_speed_t speed)
|
grub_usb_speed_t speed)
|
||||||
{
|
{
|
||||||
grub_usb_device_t dev;
|
grub_usb_device_t dev;
|
||||||
grub_err_t err;
|
grub_err_t err;
|
||||||
|
|
||||||
/* Disable the port. XXX: Why? */
|
/* Disable the port. XXX: Why? */
|
||||||
err = controller->dev->portstatus (controller, portno, 0);
|
err = hub->controller->dev->portstatus (hub->controller, portno, 0);
|
||||||
if (err)
|
if (err)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
/* Enable the port. */
|
/* Enable the port. */
|
||||||
err = controller->dev->portstatus (controller, portno, 1);
|
err = hub->controller->dev->portstatus (hub->controller, portno, 1);
|
||||||
if (err)
|
if (err)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
/* Enable the port and create a device. */
|
/* Enable the port and create a device. */
|
||||||
dev = grub_usb_hub_add_dev (controller, speed);
|
dev = grub_usb_hub_add_dev (hub->controller, speed);
|
||||||
if (! dev)
|
if (! dev)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
hub->devices[portno] = dev;
|
||||||
|
|
||||||
/* If the device is a Hub, scan it for more devices. */
|
/* If the device is a Hub, scan it for more devices. */
|
||||||
if (dev->descdev.class == 0x09)
|
if (dev->descdev.class == 0x09)
|
||||||
grub_usb_add_hub (dev);
|
grub_usb_add_hub (dev);
|
||||||
|
@ -297,8 +309,8 @@ grub_usb_root_hub (grub_usb_controller_t controller)
|
||||||
|
|
||||||
/* Query the number of ports the root Hub has. */
|
/* Query the number of ports the root Hub has. */
|
||||||
hub->nports = controller->dev->hubports (controller);
|
hub->nports = controller->dev->hubports (controller);
|
||||||
hub->speed = grub_malloc (sizeof (hub->speed[0]) * hub->nports);
|
hub->devices = grub_zalloc (sizeof (hub->devices[0]) * hub->nports);
|
||||||
if (!hub->speed)
|
if (!hub->devices)
|
||||||
{
|
{
|
||||||
grub_free (hub->controller);
|
grub_free (hub->controller);
|
||||||
grub_free (hub);
|
grub_free (hub);
|
||||||
|
@ -307,36 +319,54 @@ grub_usb_root_hub (grub_usb_controller_t controller)
|
||||||
|
|
||||||
for (i = 0; i < hub->nports; i++)
|
for (i = 0; i < hub->nports; i++)
|
||||||
{
|
{
|
||||||
hub->speed[i] = controller->dev->detect_dev (hub->controller, i,
|
grub_usb_speed_t speed;
|
||||||
|
speed = controller->dev->detect_dev (hub->controller, i,
|
||||||
&changed);
|
&changed);
|
||||||
|
|
||||||
if (hub->speed[i] != GRUB_USB_SPEED_NONE)
|
if (speed != GRUB_USB_SPEED_NONE)
|
||||||
attach_root_port (hub->controller, i, hub->speed[i]);
|
attach_root_port (hub, i, speed);
|
||||||
}
|
}
|
||||||
|
|
||||||
return GRUB_USB_ERR_NONE;
|
return GRUB_USB_ERR_NONE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void detach_device (grub_usb_device_t dev);
|
||||||
|
|
||||||
|
static void
|
||||||
|
detach_device (grub_usb_device_t dev)
|
||||||
|
{
|
||||||
|
unsigned i;
|
||||||
|
int k;
|
||||||
|
if (!dev)
|
||||||
|
return;
|
||||||
|
if (dev->descdev.class == GRUB_USB_CLASS_HUB)
|
||||||
|
{
|
||||||
|
for (i = 0; i < dev->nports; i++)
|
||||||
|
detach_device (dev->children[i]);
|
||||||
|
grub_free (dev->children);
|
||||||
|
}
|
||||||
|
for (i = 0; i < ARRAY_SIZE (dev->config); i++)
|
||||||
|
if (dev->config[i].descconf)
|
||||||
|
for (k = 0; k < dev->config[i].descconf->numif; k++)
|
||||||
|
{
|
||||||
|
struct grub_usb_interface *inter = &dev->config[i].interf[k];
|
||||||
|
if (inter && inter->detach_hook)
|
||||||
|
inter->detach_hook (dev, i, k);
|
||||||
|
}
|
||||||
|
grub_usb_devs[dev->addr] = 0;
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
poll_nonroot_hub (grub_usb_device_t dev)
|
poll_nonroot_hub (grub_usb_device_t dev)
|
||||||
{
|
{
|
||||||
struct grub_usb_usb_hubdesc hubdesc;
|
|
||||||
grub_err_t err;
|
grub_err_t err;
|
||||||
int i;
|
unsigned i;
|
||||||
grub_uint64_t timeout;
|
grub_uint64_t timeout;
|
||||||
grub_usb_device_t next_dev;
|
grub_usb_device_t next_dev;
|
||||||
|
grub_usb_device_t *attached_devices = dev->children;
|
||||||
err = grub_usb_control_msg (dev, (GRUB_USB_REQTYPE_IN
|
|
||||||
| GRUB_USB_REQTYPE_CLASS
|
|
||||||
| GRUB_USB_REQTYPE_TARGET_DEV),
|
|
||||||
GRUB_USB_REQ_GET_DESCRIPTOR,
|
|
||||||
(GRUB_USB_DESCRIPTOR_HUB << 8) | 0,
|
|
||||||
0, sizeof (hubdesc), (char *) &hubdesc);
|
|
||||||
if (err)
|
|
||||||
return;
|
|
||||||
|
|
||||||
/* Iterate over the Hub ports. */
|
/* Iterate over the Hub ports. */
|
||||||
for (i = 1; i <= hubdesc.portcnt; i++)
|
for (i = 1; i <= dev->nports; i++)
|
||||||
{
|
{
|
||||||
grub_uint32_t status;
|
grub_uint32_t status;
|
||||||
|
|
||||||
|
@ -351,6 +381,12 @@ poll_nonroot_hub (grub_usb_device_t dev)
|
||||||
if (err)
|
if (err)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
|
if (status & GRUB_USB_HUB_STATUS_C_CONNECTED)
|
||||||
|
{
|
||||||
|
detach_device (attached_devices[i-1]);
|
||||||
|
attached_devices[i - 1] = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
/* Connected and status of connection changed ? */
|
/* Connected and status of connection changed ? */
|
||||||
if ((status & GRUB_USB_HUB_STATUS_CONNECTED)
|
if ((status & GRUB_USB_HUB_STATUS_CONNECTED)
|
||||||
&& (status & GRUB_USB_HUB_STATUS_C_CONNECTED))
|
&& (status & GRUB_USB_HUB_STATUS_C_CONNECTED))
|
||||||
|
@ -417,6 +453,8 @@ poll_nonroot_hub (grub_usb_device_t dev)
|
||||||
if (! next_dev)
|
if (! next_dev)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
|
attached_devices[i - 1] = next_dev;
|
||||||
|
|
||||||
/* If the device is a Hub, scan it for more devices. */
|
/* If the device is a Hub, scan it for more devices. */
|
||||||
if (next_dev->descdev.class == 0x09)
|
if (next_dev->descdev.class == 0x09)
|
||||||
grub_usb_add_hub (next_dev);
|
grub_usb_add_hub (next_dev);
|
||||||
|
@ -434,26 +472,23 @@ grub_usb_poll_devices (void)
|
||||||
|
|
||||||
for (hub = hubs; hub; hub = hub->next)
|
for (hub = hubs; hub; hub = hub->next)
|
||||||
{
|
{
|
||||||
int changed=0;
|
|
||||||
/* Do we have to recheck number of ports? */
|
/* Do we have to recheck number of ports? */
|
||||||
/* No, it should be never changed, it should be constant. */
|
/* No, it should be never changed, it should be constant. */
|
||||||
for (i = 0; i < hub->nports; i++)
|
for (i = 0; i < hub->nports; i++)
|
||||||
{
|
{
|
||||||
grub_usb_speed_t speed;
|
grub_usb_speed_t speed;
|
||||||
|
int changed = 0;
|
||||||
|
|
||||||
speed = hub->controller->dev->detect_dev (hub->controller, i,
|
speed = hub->controller->dev->detect_dev (hub->controller, i,
|
||||||
&changed);
|
&changed);
|
||||||
|
|
||||||
if (speed != GRUB_USB_SPEED_NONE)
|
|
||||||
{
|
|
||||||
if (changed)
|
if (changed)
|
||||||
attach_root_port (hub->controller, i, speed);
|
{
|
||||||
|
detach_device (hub->devices[i]);
|
||||||
|
hub->devices[i] = NULL;
|
||||||
|
if (speed != GRUB_USB_SPEED_NONE)
|
||||||
|
attach_root_port (hub, i, speed);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* XXX: There should be also handling
|
|
||||||
* of disconnected devices. */
|
|
||||||
|
|
||||||
hub->speed[i] = speed;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -463,10 +498,8 @@ grub_usb_poll_devices (void)
|
||||||
grub_usb_device_t dev = grub_usb_devs[i];
|
grub_usb_device_t dev = grub_usb_devs[i];
|
||||||
|
|
||||||
if (dev && dev->descdev.class == 0x09)
|
if (dev && dev->descdev.class == 0x09)
|
||||||
{
|
|
||||||
poll_nonroot_hub (dev);
|
poll_nonroot_hub (dev);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
10
disk/usbms.c
10
disk/usbms.c
|
@ -65,6 +65,7 @@ typedef struct grub_usbms_dev *grub_usbms_dev_t;
|
||||||
/* FIXME: remove limit. */
|
/* FIXME: remove limit. */
|
||||||
#define MAX_USBMS_DEVICES 128
|
#define MAX_USBMS_DEVICES 128
|
||||||
static grub_usbms_dev_t grub_usbms_devices[MAX_USBMS_DEVICES];
|
static grub_usbms_dev_t grub_usbms_devices[MAX_USBMS_DEVICES];
|
||||||
|
static int first_available_slot = 0;
|
||||||
|
|
||||||
static grub_err_t
|
static grub_err_t
|
||||||
grub_usbms_reset (grub_usb_device_t dev, int interface)
|
grub_usbms_reset (grub_usb_device_t dev, int interface)
|
||||||
|
@ -96,13 +97,12 @@ grub_usbms_attach (grub_usb_device_t usbdev, int configno, int interfno)
|
||||||
unsigned curnum;
|
unsigned curnum;
|
||||||
grub_usb_err_t err;
|
grub_usb_err_t err;
|
||||||
|
|
||||||
for (curnum = 0; curnum < ARRAY_SIZE (grub_usbms_devices); curnum++)
|
if (first_available_slot == ARRAY_SIZE (grub_usbms_devices))
|
||||||
if (!grub_usbms_devices[curnum])
|
|
||||||
break;
|
|
||||||
|
|
||||||
if (curnum == ARRAY_SIZE (grub_usbms_devices))
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
|
curnum = first_available_slot;
|
||||||
|
first_available_slot++;
|
||||||
|
|
||||||
interf = usbdev->config[configno].interf[interfno].descif;
|
interf = usbdev->config[configno].interf[interfno].descif;
|
||||||
|
|
||||||
if ((interf->subclass != GRUB_USBMS_SUBCLASS_BULK
|
if ((interf->subclass != GRUB_USBMS_SUBCLASS_BULK
|
||||||
|
|
|
@ -459,6 +459,7 @@ grub_print_spaces (struct grub_term_output *term, int number_spaces)
|
||||||
grub_putcode (' ', term);
|
grub_putcode (' ', term);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
extern void (*EXPORT_VAR (grub_term_poll_usb)) (void);
|
||||||
|
|
||||||
/* For convenience. */
|
/* For convenience. */
|
||||||
#define GRUB_TERM_ASCII_CHAR(c) ((c) & 0xff)
|
#define GRUB_TERM_ASCII_CHAR(c) ((c) & 0xff)
|
||||||
|
|
|
@ -177,6 +177,12 @@ struct grub_usb_device
|
||||||
|
|
||||||
/* Data toggle values (used for bulk transfers only). */
|
/* Data toggle values (used for bulk transfers only). */
|
||||||
int toggle[256];
|
int toggle[256];
|
||||||
|
|
||||||
|
/* Array of children for a hub. */
|
||||||
|
grub_usb_device_t *children;
|
||||||
|
|
||||||
|
/* Number of hub ports. */
|
||||||
|
unsigned nports;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
11
kern/term.c
11
kern/term.c
|
@ -28,6 +28,8 @@ struct grub_term_input *grub_term_inputs_disabled;
|
||||||
struct grub_term_output *grub_term_outputs;
|
struct grub_term_output *grub_term_outputs;
|
||||||
struct grub_term_input *grub_term_inputs;
|
struct grub_term_input *grub_term_inputs;
|
||||||
|
|
||||||
|
void (*grub_term_poll_usb) (void) = NULL;
|
||||||
|
|
||||||
/* Put a Unicode character. */
|
/* Put a Unicode character. */
|
||||||
static void
|
static void
|
||||||
grub_putcode_dumb (grub_uint32_t code,
|
grub_putcode_dumb (grub_uint32_t code,
|
||||||
|
@ -85,6 +87,9 @@ grub_getkey (void)
|
||||||
|
|
||||||
while (1)
|
while (1)
|
||||||
{
|
{
|
||||||
|
if (grub_term_poll_usb)
|
||||||
|
grub_term_poll_usb ();
|
||||||
|
|
||||||
FOR_ACTIVE_TERM_INPUTS(term)
|
FOR_ACTIVE_TERM_INPUTS(term)
|
||||||
{
|
{
|
||||||
int key = term->checkkey (term);
|
int key = term->checkkey (term);
|
||||||
|
@ -101,6 +106,9 @@ grub_checkkey (void)
|
||||||
{
|
{
|
||||||
grub_term_input_t term;
|
grub_term_input_t term;
|
||||||
|
|
||||||
|
if (grub_term_poll_usb)
|
||||||
|
grub_term_poll_usb ();
|
||||||
|
|
||||||
FOR_ACTIVE_TERM_INPUTS(term)
|
FOR_ACTIVE_TERM_INPUTS(term)
|
||||||
{
|
{
|
||||||
int key = term->checkkey (term);
|
int key = term->checkkey (term);
|
||||||
|
@ -117,6 +125,9 @@ grub_getkeystatus (void)
|
||||||
int status = 0;
|
int status = 0;
|
||||||
grub_term_input_t term;
|
grub_term_input_t term;
|
||||||
|
|
||||||
|
if (grub_term_poll_usb)
|
||||||
|
grub_term_poll_usb ();
|
||||||
|
|
||||||
FOR_ACTIVE_TERM_INPUTS(term)
|
FOR_ACTIVE_TERM_INPUTS(term)
|
||||||
{
|
{
|
||||||
if (term->getkeystatus)
|
if (term->getkeystatus)
|
||||||
|
|
|
@ -95,8 +95,15 @@ grub_usb_keyboard_detach (grub_usb_device_t usbdev,
|
||||||
{
|
{
|
||||||
unsigned i;
|
unsigned i;
|
||||||
for (i = 0; i < ARRAY_SIZE (grub_usb_keyboards); i++)
|
for (i = 0; i < ARRAY_SIZE (grub_usb_keyboards); i++)
|
||||||
if (grub_usb_keyboards[i].data && grub_usb_keyboards[i].data == usbdev)
|
|
||||||
{
|
{
|
||||||
|
struct grub_usb_keyboard_data *data = grub_usb_keyboards[i].data;
|
||||||
|
|
||||||
|
if (!data)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (data->usbdev != usbdev)
|
||||||
|
continue;
|
||||||
|
|
||||||
grub_term_unregister_input (&grub_usb_keyboards[i]);
|
grub_term_unregister_input (&grub_usb_keyboards[i]);
|
||||||
grub_free ((char *) grub_usb_keyboards[i].name);
|
grub_free ((char *) grub_usb_keyboards[i].name);
|
||||||
grub_usb_keyboards[i].name = NULL;
|
grub_usb_keyboards[i].name = NULL;
|
||||||
|
|
Loading…
Reference in a new issue