2009-02-08 Marco Gerards <marco@gnu.org>
* Makefile.in (enable_grub_emu_usb): New variable.
* conf/i386-pc.rmk (grub_emu_SOURCES): Add `disk/scsi.c'.
(grub_emu_SOURCES) [grub_emu_SOURCES]: Add `disk/usbms.c',
`util/usb.c', `bus/usb/usb.c' and `commands/usbtest.c'.
(grub_emu_LDFLAGS): Add `$(LIBUSB)'.
(pkglib_MODULES): Add `usb.mod', `uhci.mod', `ohci.mod',
`usbtest.mod' and `usbms.mod'.
(usb_mod_SOURCES, usb_mod_CFLAGS, usb_mod_LDFLAGS)
(usbtest_mod_SOURCES, usbtest_mod_CFLAGS, usbtest_mod_LDFLAGS)
(uhci_mod_SOURCES, uhci_mod_CFLAGS, uhci_mod_LDFLAGS,
(ohci_mod_SOURCES, ohci_mod_CFLAGS, ohci_mod_LDFLAGS)
(usbms_mod_SOURCES, usbms_mod_CFLAGS, usbms_mod_LDFLAGS): New
variables.
* disk/usbms.c: New file.
* include/grub/usb.h: Likewise.
* include/grub/usbtrans.h: Likewise.
* include/grub/usbdesc.h: Likewise.
* bus/usb/usbtrans.c: Likewise.
* bus/usb/ohci.c: Likewise.
* bus/usb/uhci.c: Likewise.
* bus/usb/usbhub.c: Likewise.
* bus/usb/usb.c: Likewise.
* commands/usbtest.c: Likewise.
* util/usb.c: Likewise.
* include/grub/err.h (grub_err_t): Add `GRUB_ERR_IO'.
* configure.ac: Test for libusb presence.
* util/grub-emu.c (main) [HAVE_LIBUSB_H]: Call `grub_libusb_init'.
2009-02-08 17:58:32 +00:00
|
|
|
|
/* usb.c -- libusb USB support for GRUB. */
|
|
|
|
|
/*
|
|
|
|
|
* GRUB -- GRand Unified Bootloader
|
|
|
|
|
* Copyright (C) 2008 Free Software Foundation, Inc.
|
|
|
|
|
*
|
|
|
|
|
* GRUB is free software: you can redistribute it and/or modify
|
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
|
* (at your option) any later version.
|
|
|
|
|
*
|
|
|
|
|
* GRUB is distributed in the hope that it will be useful,
|
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
|
*
|
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
|
* along with GRUB. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
#include <grub/misc.h>
|
|
|
|
|
#include <grub/mm.h>
|
|
|
|
|
#include <usb.h>
|
|
|
|
|
#include <grub/usb.h>
|
|
|
|
|
#include <grub/dl.h>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static struct grub_usb_controller_dev usb_controller =
|
|
|
|
|
{
|
|
|
|
|
.name = "libusb"
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static struct grub_usb_device *grub_usb_devs[128];
|
|
|
|
|
|
|
|
|
|
struct usb_bus *busses;
|
|
|
|
|
|
|
|
|
|
static grub_err_t
|
|
|
|
|
grub_libusb_devices (void)
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
struct usb_bus *bus;
|
|
|
|
|
int last = 0;
|
|
|
|
|
|
|
|
|
|
busses = usb_get_busses();
|
|
|
|
|
|
|
|
|
|
for (bus = busses; bus; bus = bus->next)
|
|
|
|
|
{
|
|
|
|
|
struct usb_device *usbdev;
|
|
|
|
|
struct grub_usb_device *dev;
|
|
|
|
|
|
|
|
|
|
for (usbdev = bus->devices; usbdev; usbdev = usbdev->next)
|
|
|
|
|
{
|
|
|
|
|
struct usb_device_descriptor *desc = &usbdev->descriptor;
|
|
|
|
|
|
|
|
|
|
if (! desc->bcdUSB)
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
dev = grub_malloc (sizeof (*dev));
|
|
|
|
|
if (! dev)
|
|
|
|
|
return grub_errno;
|
|
|
|
|
|
|
|
|
|
dev->data = usbdev;
|
|
|
|
|
|
|
|
|
|
/* Fill in all descriptors. */
|
|
|
|
|
grub_usb_device_initialize (dev);
|
|
|
|
|
|
|
|
|
|
/* Register the device. */
|
|
|
|
|
grub_usb_devs[last++] = dev;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return GRUB_USB_ERR_NONE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
grub_err_t
|
|
|
|
|
grub_libusb_init (void)
|
|
|
|
|
{
|
|
|
|
|
usb_init();
|
|
|
|
|
usb_find_busses();
|
|
|
|
|
usb_find_devices();
|
|
|
|
|
|
|
|
|
|
if (grub_libusb_devices ())
|
|
|
|
|
return grub_errno;
|
|
|
|
|
|
2009-06-10 21:04:23 +00:00
|
|
|
|
grub_usb_controller_dev_register (&usb_controller);
|
2009-02-08 Marco Gerards <marco@gnu.org>
* Makefile.in (enable_grub_emu_usb): New variable.
* conf/i386-pc.rmk (grub_emu_SOURCES): Add `disk/scsi.c'.
(grub_emu_SOURCES) [grub_emu_SOURCES]: Add `disk/usbms.c',
`util/usb.c', `bus/usb/usb.c' and `commands/usbtest.c'.
(grub_emu_LDFLAGS): Add `$(LIBUSB)'.
(pkglib_MODULES): Add `usb.mod', `uhci.mod', `ohci.mod',
`usbtest.mod' and `usbms.mod'.
(usb_mod_SOURCES, usb_mod_CFLAGS, usb_mod_LDFLAGS)
(usbtest_mod_SOURCES, usbtest_mod_CFLAGS, usbtest_mod_LDFLAGS)
(uhci_mod_SOURCES, uhci_mod_CFLAGS, uhci_mod_LDFLAGS,
(ohci_mod_SOURCES, ohci_mod_CFLAGS, ohci_mod_LDFLAGS)
(usbms_mod_SOURCES, usbms_mod_CFLAGS, usbms_mod_LDFLAGS): New
variables.
* disk/usbms.c: New file.
* include/grub/usb.h: Likewise.
* include/grub/usbtrans.h: Likewise.
* include/grub/usbdesc.h: Likewise.
* bus/usb/usbtrans.c: Likewise.
* bus/usb/ohci.c: Likewise.
* bus/usb/uhci.c: Likewise.
* bus/usb/usbhub.c: Likewise.
* bus/usb/usb.c: Likewise.
* commands/usbtest.c: Likewise.
* util/usb.c: Likewise.
* include/grub/err.h (grub_err_t): Add `GRUB_ERR_IO'.
* configure.ac: Test for libusb presence.
* util/grub-emu.c (main) [HAVE_LIBUSB_H]: Call `grub_libusb_init'.
2009-02-08 17:58:32 +00:00
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
grub_err_t
|
|
|
|
|
grub_libusb_fini (void)
|
|
|
|
|
{
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
grub_usb_iterate (int (*hook) (grub_usb_device_t dev))
|
|
|
|
|
{
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < 128; i++)
|
|
|
|
|
{
|
|
|
|
|
if (grub_usb_devs[i])
|
|
|
|
|
{
|
|
|
|
|
if (hook (grub_usb_devs[i]))
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
grub_usb_err_t
|
|
|
|
|
grub_usb_root_hub (grub_usb_controller_t controller __attribute__((unused)))
|
|
|
|
|
{
|
|
|
|
|
return GRUB_USB_ERR_NONE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
grub_usb_err_t
|
|
|
|
|
grub_usb_control_msg (grub_usb_device_t dev, grub_uint8_t reqtype,
|
|
|
|
|
grub_uint8_t request, grub_uint16_t value,
|
|
|
|
|
grub_uint16_t index, grub_size_t size, char *data)
|
|
|
|
|
{
|
|
|
|
|
usb_dev_handle *devh;
|
|
|
|
|
struct usb_device *d = dev->data;
|
|
|
|
|
|
|
|
|
|
devh = usb_open (d);
|
|
|
|
|
if (usb_control_msg (devh, reqtype, request,
|
|
|
|
|
value, index, data, size, 20) < 0)
|
|
|
|
|
{
|
|
|
|
|
usb_close (devh);
|
|
|
|
|
return GRUB_USB_ERR_STALL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
usb_close (devh);
|
|
|
|
|
|
|
|
|
|
return GRUB_USB_ERR_NONE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
grub_usb_err_t
|
|
|
|
|
grub_usb_bulk_read (grub_usb_device_t dev,
|
|
|
|
|
int endpoint, grub_size_t size, char *data)
|
|
|
|
|
{
|
|
|
|
|
usb_dev_handle *devh;
|
|
|
|
|
struct usb_device *d = dev->data;
|
|
|
|
|
|
|
|
|
|
devh = usb_open (d);
|
|
|
|
|
if (usb_claim_interface (devh, 0) < 1)
|
|
|
|
|
{
|
|
|
|
|
usb_close (devh);
|
|
|
|
|
return GRUB_USB_ERR_STALL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (usb_bulk_read (devh, endpoint, data, size, 20) < 1)
|
|
|
|
|
{
|
|
|
|
|
usb_close (devh);
|
|
|
|
|
return GRUB_USB_ERR_STALL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
usb_release_interface (devh, 0);
|
|
|
|
|
usb_close (devh);
|
|
|
|
|
|
|
|
|
|
return GRUB_USB_ERR_NONE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
grub_usb_err_t
|
|
|
|
|
grub_usb_bulk_write (grub_usb_device_t dev,
|
|
|
|
|
int endpoint, grub_size_t size, char *data)
|
|
|
|
|
{
|
|
|
|
|
usb_dev_handle *devh;
|
|
|
|
|
struct usb_device *d = dev->data;
|
|
|
|
|
|
|
|
|
|
devh = usb_open (d);
|
|
|
|
|
if (usb_claim_interface (devh, 0) < 0)
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (usb_bulk_write (devh, endpoint, data, size, 20) < 0)
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
if (usb_release_interface (devh, 0) < 0)
|
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
|
|
usb_close (devh);
|
|
|
|
|
|
|
|
|
|
return GRUB_USB_ERR_NONE;
|
|
|
|
|
|
|
|
|
|
fail:
|
|
|
|
|
usb_close (devh);
|
|
|
|
|
return GRUB_USB_ERR_STALL;
|
|
|
|
|
}
|