separate function to allocate DMA memory
This commit is contained in:
parent
469ed9410b
commit
839ced08f5
7 changed files with 167 additions and 49 deletions
43
bus/pci.c
43
bus/pci.c
|
@ -19,6 +19,49 @@
|
||||||
|
|
||||||
#include <grub/dl.h>
|
#include <grub/dl.h>
|
||||||
#include <grub/pci.h>
|
#include <grub/pci.h>
|
||||||
|
#include <grub/mm.h>
|
||||||
|
|
||||||
|
#if GRUB_TARGET_SIZEOF_VOID_P == 4
|
||||||
|
struct grub_pci_dma_chunk *
|
||||||
|
grub_memalign_dma32 (grub_size_t align, grub_size_t size)
|
||||||
|
{
|
||||||
|
return grub_memalign (align, size);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
grub_dma_free (struct grub_pci_dma_chunk *ch)
|
||||||
|
{
|
||||||
|
grub_free (ch);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef GRUB_MACHINE_MIPS_YEELOONG
|
||||||
|
void *
|
||||||
|
grub_dma_get_virt (struct grub_pci_dma_chunk *ch)
|
||||||
|
{
|
||||||
|
return (void *) ((((grub_uint32_t) ch) & 0x1fffffff) | 0xa0000000);
|
||||||
|
}
|
||||||
|
|
||||||
|
grub_uint32_t
|
||||||
|
grub_dma_get_phys (struct grub_pci_dma_chunk *ch)
|
||||||
|
{
|
||||||
|
return ((grub_uint32_t) ch) & 0x1fffffff;
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
|
||||||
|
void *
|
||||||
|
grub_dma_get_virt (struct grub_pci_dma_chunk *ch)
|
||||||
|
{
|
||||||
|
return (void *) ch;
|
||||||
|
}
|
||||||
|
|
||||||
|
grub_uint32_t
|
||||||
|
grub_dma_get_phys (struct grub_pci_dma_chunk *ch)
|
||||||
|
{
|
||||||
|
return (grub_uint32_t) ch;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
grub_pci_address_t
|
grub_pci_address_t
|
||||||
grub_pci_make_address (grub_pci_device_t dev, int reg)
|
grub_pci_make_address (grub_pci_device_t dev, int reg)
|
||||||
|
|
|
@ -72,6 +72,7 @@ struct grub_ohci
|
||||||
volatile grub_uint32_t *iobase;
|
volatile grub_uint32_t *iobase;
|
||||||
volatile struct grub_ohci_hcca *hcca;
|
volatile struct grub_ohci_hcca *hcca;
|
||||||
grub_uint32_t hcca_addr;
|
grub_uint32_t hcca_addr;
|
||||||
|
struct grub_pci_dma_chunk *hcca_chunk;
|
||||||
struct grub_ohci *next;
|
struct grub_ohci *next;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -136,7 +137,6 @@ grub_ohci_pci_iter (grub_pci_device_t dev,
|
||||||
grub_uint32_t revision;
|
grub_uint32_t revision;
|
||||||
grub_uint32_t frame_interval;
|
grub_uint32_t frame_interval;
|
||||||
int cs5536;
|
int cs5536;
|
||||||
grub_uint32_t hcca_addr;
|
|
||||||
|
|
||||||
/* Determine IO base address. */
|
/* Determine IO base address. */
|
||||||
grub_dprintf ("ohci", "pciid = %x\n", pciid);
|
grub_dprintf ("ohci", "pciid = %x\n", pciid);
|
||||||
|
@ -157,7 +157,6 @@ grub_ohci_pci_iter (grub_pci_device_t dev,
|
||||||
basereg &= ~GRUB_CS5536_MSR_USB_BASE_PME_ENABLED;
|
basereg &= ~GRUB_CS5536_MSR_USB_BASE_PME_ENABLED;
|
||||||
basereg &= ~GRUB_CS5536_MSR_USB_BASE_PME_STATUS;
|
basereg &= ~GRUB_CS5536_MSR_USB_BASE_PME_STATUS;
|
||||||
grub_cs5536_write_msr (dev, GRUB_CS5536_MSR_USB_OHCI_BASE, basereg);
|
grub_cs5536_write_msr (dev, GRUB_CS5536_MSR_USB_OHCI_BASE, basereg);
|
||||||
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
@ -198,14 +197,12 @@ grub_ohci_pci_iter (grub_pci_device_t dev,
|
||||||
|
|
||||||
grub_dprintf ("ohci", "base=%p\n", o->iobase);
|
grub_dprintf ("ohci", "base=%p\n", o->iobase);
|
||||||
|
|
||||||
/* FIXME: create proper abstraction for this. */
|
|
||||||
#ifdef GRUB_MACHINE_MIPS_YEELOONG
|
|
||||||
hcca_addr = 0x05000100;
|
|
||||||
#else
|
|
||||||
/* Reserve memory for the HCCA. */
|
/* Reserve memory for the HCCA. */
|
||||||
hcca_addr = (grub_uint32_t) grub_memalign (256, 256);
|
o->hcca_chunk = grub_memalign_dma32 (256, 256);
|
||||||
#endif
|
if (! o->hcca_chunk)
|
||||||
o->hcca = grub_pci_device_map_range (dev, hcca_addr, 256);
|
return 1;
|
||||||
|
o->hcca = grub_dma_get_virt (o->hcca_chunk);
|
||||||
|
o->hcca_addr = grub_dma_get_phys (o->hcca_chunk);
|
||||||
|
|
||||||
/* Check if the OHCI revision is actually 1.0 as supported. */
|
/* Check if the OHCI revision is actually 1.0 as supported. */
|
||||||
revision = grub_ohci_readreg32 (o, GRUB_OHCI_REG_REVISION);
|
revision = grub_ohci_readreg32 (o, GRUB_OHCI_REG_REVISION);
|
||||||
|
@ -236,7 +233,7 @@ grub_ohci_pci_iter (grub_pci_device_t dev,
|
||||||
GRUB_OHCI_PERIODIC_START);
|
GRUB_OHCI_PERIODIC_START);
|
||||||
|
|
||||||
/* Setup the HCCA. */
|
/* Setup the HCCA. */
|
||||||
grub_ohci_writereg32 (o, GRUB_OHCI_REG_HCCA, hcca_addr);
|
grub_ohci_writereg32 (o, GRUB_OHCI_REG_HCCA, o->hcca_addr);
|
||||||
grub_dprintf ("ohci", "OHCI HCCA\n");
|
grub_dprintf ("ohci", "OHCI HCCA\n");
|
||||||
|
|
||||||
/* Enable the OHCI. */
|
/* Enable the OHCI. */
|
||||||
|
@ -289,7 +286,7 @@ grub_ohci_iterate (int (*hook) (grub_usb_controller_t dev))
|
||||||
static void
|
static void
|
||||||
grub_ohci_transaction (grub_ohci_td_t td,
|
grub_ohci_transaction (grub_ohci_td_t td,
|
||||||
grub_transfer_type_t type, unsigned int toggle,
|
grub_transfer_type_t type, unsigned int toggle,
|
||||||
grub_size_t size, char *data)
|
grub_size_t size, grub_uint32_t data)
|
||||||
{
|
{
|
||||||
grub_uint32_t token;
|
grub_uint32_t token;
|
||||||
grub_uint32_t buffer;
|
grub_uint32_t buffer;
|
||||||
|
@ -321,7 +318,7 @@ grub_ohci_transaction (grub_ohci_td_t td,
|
||||||
token |= toggle << 24;
|
token |= toggle << 24;
|
||||||
token |= 1 << 25;
|
token |= 1 << 25;
|
||||||
|
|
||||||
buffer = (grub_uint32_t) data;
|
buffer = data;
|
||||||
buffer_end = buffer + size - 1;
|
buffer_end = buffer + size - 1;
|
||||||
|
|
||||||
td->token = grub_cpu_to_le32 (token);
|
td->token = grub_cpu_to_le32 (token);
|
||||||
|
@ -336,7 +333,10 @@ grub_ohci_transfer (grub_usb_controller_t dev,
|
||||||
{
|
{
|
||||||
struct grub_ohci *o = (struct grub_ohci *) dev->data;
|
struct grub_ohci *o = (struct grub_ohci *) dev->data;
|
||||||
grub_ohci_ed_t ed;
|
grub_ohci_ed_t ed;
|
||||||
|
grub_uint32_t ed_addr;
|
||||||
|
struct grub_pci_dma_chunk *ed_chunk, *td_list_chunk;
|
||||||
grub_ohci_td_t td_list;
|
grub_ohci_td_t td_list;
|
||||||
|
grub_uint32_t td_list_addr;
|
||||||
grub_uint32_t target;
|
grub_uint32_t target;
|
||||||
grub_uint32_t td_tail;
|
grub_uint32_t td_tail;
|
||||||
grub_uint32_t td_head;
|
grub_uint32_t td_head;
|
||||||
|
@ -346,18 +346,23 @@ grub_ohci_transfer (grub_usb_controller_t dev,
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
/* Allocate an Endpoint Descriptor. */
|
/* Allocate an Endpoint Descriptor. */
|
||||||
ed = grub_memalign (16, sizeof (*ed));
|
ed_chunk = grub_memalign_dma32 (256, sizeof (*ed));
|
||||||
if (! ed)
|
if (! ed_chunk)
|
||||||
return GRUB_USB_ERR_INTERNAL;
|
return GRUB_USB_ERR_INTERNAL;
|
||||||
|
ed = grub_dma_get_virt (ed_chunk);
|
||||||
|
ed_addr = grub_dma_get_phys (ed_chunk);
|
||||||
|
|
||||||
td_list = grub_memalign (16, sizeof (*td_list) * (transfer->transcnt + 1));
|
td_list_chunk = grub_memalign_dma32 (256, sizeof (*td_list)
|
||||||
if (! td_list)
|
* (transfer->transcnt + 1));
|
||||||
|
if (! td_list_chunk)
|
||||||
{
|
{
|
||||||
grub_free ((void *) ed);
|
grub_dma_free (ed_chunk);
|
||||||
return GRUB_USB_ERR_INTERNAL;
|
return GRUB_USB_ERR_INTERNAL;
|
||||||
}
|
}
|
||||||
|
td_list = grub_dma_get_virt (td_list_chunk);
|
||||||
|
td_list_addr = grub_dma_get_phys (td_list_chunk);
|
||||||
|
|
||||||
grub_dprintf ("ohci", "alloc=%p\n", td_list);
|
grub_dprintf ("ohci", "alloc=%p/0x%x\n", td_list, td_list_addr);
|
||||||
|
|
||||||
/* Setup all Transfer Descriptors. */
|
/* Setup all Transfer Descriptors. */
|
||||||
for (i = 0; i < transfer->transcnt; i++)
|
for (i = 0; i < transfer->transcnt; i++)
|
||||||
|
@ -367,7 +372,8 @@ grub_ohci_transfer (grub_usb_controller_t dev,
|
||||||
grub_ohci_transaction (&td_list[i], tr->pid, tr->toggle,
|
grub_ohci_transaction (&td_list[i], tr->pid, tr->toggle,
|
||||||
tr->size, tr->data);
|
tr->size, tr->data);
|
||||||
|
|
||||||
td_list[i].next_td = grub_cpu_to_le32 (&td_list[i + 1]);
|
td_list[i].next_td = grub_cpu_to_le32 (td_list_addr
|
||||||
|
+ (i + 1) * sizeof (td_list[0]));
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Setup the Endpoint Descriptor. */
|
/* Setup the Endpoint Descriptor. */
|
||||||
|
@ -384,9 +390,9 @@ grub_ohci_transfer (grub_usb_controller_t dev,
|
||||||
/* Set the maximum packet size. */
|
/* Set the maximum packet size. */
|
||||||
target |= transfer->max << 16;
|
target |= transfer->max << 16;
|
||||||
|
|
||||||
td_head = (grub_uint32_t) td_list;
|
td_head = td_list_addr;
|
||||||
|
|
||||||
td_tail = (grub_uint32_t) &td_list[transfer->transcnt];
|
td_tail = td_list_addr + transfer->transcnt * sizeof (*td_list);
|
||||||
|
|
||||||
ed->target = grub_cpu_to_le32 (target);
|
ed->target = grub_cpu_to_le32 (target);
|
||||||
ed->td_head = grub_cpu_to_le32 (td_head);
|
ed->td_head = grub_cpu_to_le32 (td_head);
|
||||||
|
@ -413,7 +419,7 @@ grub_ohci_transfer (grub_usb_controller_t dev,
|
||||||
status &= ~(1 << 2);
|
status &= ~(1 << 2);
|
||||||
grub_ohci_writereg32 (o, GRUB_OHCI_REG_CMDSTATUS, status);
|
grub_ohci_writereg32 (o, GRUB_OHCI_REG_CMDSTATUS, status);
|
||||||
|
|
||||||
grub_ohci_writereg32 (o, GRUB_OHCI_REG_BULKHEAD, (grub_uint32_t) ed);
|
grub_ohci_writereg32 (o, GRUB_OHCI_REG_BULKHEAD, ed_addr);
|
||||||
|
|
||||||
/* Enable the Bulk list. */
|
/* Enable the Bulk list. */
|
||||||
control |= 1 << 5;
|
control |= 1 << 5;
|
||||||
|
@ -440,10 +446,9 @@ grub_ohci_transfer (grub_usb_controller_t dev,
|
||||||
status &= ~(1 << 1);
|
status &= ~(1 << 1);
|
||||||
grub_ohci_writereg32 (o, GRUB_OHCI_REG_CMDSTATUS, status);
|
grub_ohci_writereg32 (o, GRUB_OHCI_REG_CMDSTATUS, status);
|
||||||
|
|
||||||
grub_ohci_writereg32 (o, GRUB_OHCI_REG_CONTROLHEAD,
|
grub_ohci_writereg32 (o, GRUB_OHCI_REG_CONTROLHEAD, ed_addr);
|
||||||
(grub_uint32_t) ed);
|
|
||||||
grub_ohci_writereg32 (o, GRUB_OHCI_REG_CONTROLHEAD+1,
|
grub_ohci_writereg32 (o, GRUB_OHCI_REG_CONTROLHEAD+1,
|
||||||
(grub_uint32_t) ed);
|
ed_addr);
|
||||||
|
|
||||||
/* Enable the Control list. */
|
/* Enable the Control list. */
|
||||||
control |= 1 << 4;
|
control |= 1 << 4;
|
||||||
|
@ -484,9 +489,12 @@ grub_ohci_transfer (grub_usb_controller_t dev,
|
||||||
{
|
{
|
||||||
grub_uint8_t errcode;
|
grub_uint8_t errcode;
|
||||||
grub_ohci_td_t tderr;
|
grub_ohci_td_t tderr;
|
||||||
|
grub_uint32_t td_err_addr;
|
||||||
|
|
||||||
tderr = (grub_ohci_td_t) grub_ohci_readreg32 (o,
|
td_err_addr = grub_ohci_readreg32 (o, GRUB_OHCI_REG_DONEHEAD);
|
||||||
GRUB_OHCI_REG_DONEHEAD);
|
|
||||||
|
tderr = (grub_ohci_td_t) ((char *) td_list
|
||||||
|
+ (td_err_addr - td_list_addr));
|
||||||
errcode = tderr->token >> 28;
|
errcode = tderr->token >> 28;
|
||||||
|
|
||||||
switch (errcode)
|
switch (errcode)
|
||||||
|
|
|
@ -380,7 +380,7 @@ static grub_uhci_td_t
|
||||||
grub_uhci_transaction (struct grub_uhci *u, unsigned int endp,
|
grub_uhci_transaction (struct grub_uhci *u, unsigned int endp,
|
||||||
grub_transfer_type_t type, unsigned int addr,
|
grub_transfer_type_t type, unsigned int addr,
|
||||||
unsigned int toggle, grub_size_t size,
|
unsigned int toggle, grub_size_t size,
|
||||||
char *data)
|
grub_uint32_t data)
|
||||||
{
|
{
|
||||||
grub_uhci_td_t td;
|
grub_uhci_td_t td;
|
||||||
static const unsigned int tf[] = { 0x69, 0xE1, 0x2D };
|
static const unsigned int tf[] = { 0x69, 0xE1, 0x2D };
|
||||||
|
@ -398,7 +398,7 @@ grub_uhci_transaction (struct grub_uhci *u, unsigned int endp,
|
||||||
}
|
}
|
||||||
|
|
||||||
grub_dprintf ("uhci",
|
grub_dprintf ("uhci",
|
||||||
"transaction: endp=%d, type=%d, addr=%d, toggle=%d, size=%d data=%p td=%p\n",
|
"transaction: endp=%d, type=%d, addr=%d, toggle=%d, size=%d data=0x%x td=%p\n",
|
||||||
endp, type, addr, toggle, size, data, td);
|
endp, type, addr, toggle, size, data, td);
|
||||||
|
|
||||||
/* Don't point to any TD, just terminate. */
|
/* Don't point to any TD, just terminate. */
|
||||||
|
@ -418,7 +418,7 @@ grub_uhci_transaction (struct grub_uhci *u, unsigned int endp,
|
||||||
td->token = ((size << 21) | (toggle << 19) | (endp << 15)
|
td->token = ((size << 21) | (toggle << 19) | (endp << 15)
|
||||||
| (addr << 8) | tf[type]);
|
| (addr << 8) | tf[type]);
|
||||||
|
|
||||||
td->buffer = (grub_uint32_t) data;
|
td->buffer = data;
|
||||||
|
|
||||||
return td;
|
return td;
|
||||||
}
|
}
|
||||||
|
|
|
@ -18,6 +18,7 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <grub/dl.h>
|
#include <grub/dl.h>
|
||||||
|
#include <grub/pci.h>
|
||||||
#include <grub/mm.h>
|
#include <grub/mm.h>
|
||||||
#include <grub/misc.h>
|
#include <grub/misc.h>
|
||||||
#include <grub/usb.h>
|
#include <grub/usb.h>
|
||||||
|
@ -29,30 +30,59 @@ grub_usb_control_msg (grub_usb_device_t dev,
|
||||||
grub_uint8_t request,
|
grub_uint8_t request,
|
||||||
grub_uint16_t value,
|
grub_uint16_t value,
|
||||||
grub_uint16_t index,
|
grub_uint16_t index,
|
||||||
grub_size_t size, char *data)
|
grub_size_t size0, char *data_in)
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
grub_usb_transfer_t transfer;
|
grub_usb_transfer_t transfer;
|
||||||
int datablocks;
|
int datablocks;
|
||||||
struct grub_usb_packet_setup setupdata;
|
struct grub_usb_packet_setup *setupdata;
|
||||||
|
grub_uint32_t setupdata_addr;
|
||||||
grub_usb_err_t err;
|
grub_usb_err_t err;
|
||||||
unsigned int max;
|
unsigned int max;
|
||||||
|
struct grub_pci_dma_chunk *data_chunk, *setupdata_chunk;
|
||||||
|
char *data;
|
||||||
|
grub_uint32_t data_addr;
|
||||||
|
grub_size_t size = size0;
|
||||||
|
|
||||||
|
/* FIXME: avoid allocation any kind of buffer in a first place. */
|
||||||
|
data_chunk = grub_memalign_dma32 (128, size ? : 16);
|
||||||
|
if (!data_chunk)
|
||||||
|
return GRUB_USB_ERR_INTERNAL;
|
||||||
|
data = grub_dma_get_virt (data_chunk);
|
||||||
|
data_addr = grub_dma_get_phys (data_chunk);
|
||||||
|
grub_memcpy (data, data_in, size);
|
||||||
|
|
||||||
grub_dprintf ("usb",
|
grub_dprintf ("usb",
|
||||||
"control: reqtype=0x%02x req=0x%02x val=0x%02x idx=0x%02x size=%d\n",
|
"control: reqtype=0x%02x req=0x%02x val=0x%02x idx=0x%02x size=%d\n",
|
||||||
reqtype, request, value, index, size);
|
reqtype, request, value, index, size);
|
||||||
|
|
||||||
/* Create a transfer. */
|
/* Create a transfer. */
|
||||||
transfer = grub_malloc (sizeof (struct grub_usb_transfer));
|
transfer = grub_malloc (sizeof (*transfer));
|
||||||
if (! transfer)
|
if (! transfer)
|
||||||
return grub_errno;
|
{
|
||||||
|
grub_dma_free (data_chunk);
|
||||||
|
return grub_errno;
|
||||||
|
}
|
||||||
|
|
||||||
|
setupdata_chunk = grub_memalign_dma32 (32, sizeof (*setupdata));
|
||||||
|
if (! setupdata_chunk)
|
||||||
|
{
|
||||||
|
grub_free (transfer);
|
||||||
|
grub_dma_free (data_chunk);
|
||||||
|
return grub_errno;
|
||||||
|
}
|
||||||
|
|
||||||
|
setupdata = grub_dma_get_virt (setupdata_chunk);
|
||||||
|
setupdata_addr = grub_dma_get_phys (setupdata_chunk);
|
||||||
|
|
||||||
/* Determine the maximum packet size. */
|
/* Determine the maximum packet size. */
|
||||||
if (dev->initialized)
|
if (dev->initialized && dev->descdev.maxsize0)
|
||||||
max = dev->descdev.maxsize0;
|
max = dev->descdev.maxsize0;
|
||||||
else
|
else
|
||||||
max = 64;
|
max = 64;
|
||||||
|
|
||||||
|
grub_dprintf ("usb", "transfer = %p, dev = %p\n", transfer, dev);
|
||||||
|
|
||||||
datablocks = (size + max - 1) / max;
|
datablocks = (size + max - 1) / max;
|
||||||
|
|
||||||
/* XXX: Discriminate between different types of control
|
/* XXX: Discriminate between different types of control
|
||||||
|
@ -71,18 +101,20 @@ grub_usb_control_msg (grub_usb_device_t dev,
|
||||||
if (! transfer->transactions)
|
if (! transfer->transactions)
|
||||||
{
|
{
|
||||||
grub_free (transfer);
|
grub_free (transfer);
|
||||||
|
grub_dma_free (setupdata_chunk);
|
||||||
|
grub_dma_free (data_chunk);
|
||||||
return grub_errno;
|
return grub_errno;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Build a Setup packet. XXX: Endianness. */
|
/* Build a Setup packet. XXX: Endianness. */
|
||||||
setupdata.reqtype = reqtype;
|
setupdata->reqtype = reqtype;
|
||||||
setupdata.request = request;
|
setupdata->request = request;
|
||||||
setupdata.value = value;
|
setupdata->value = value;
|
||||||
setupdata.index = index;
|
setupdata->index = index;
|
||||||
setupdata.length = size;
|
setupdata->length = size;
|
||||||
transfer->transactions[0].size = sizeof (setupdata);
|
transfer->transactions[0].size = sizeof (*setupdata);
|
||||||
transfer->transactions[0].pid = GRUB_USB_TRANSFER_TYPE_SETUP;
|
transfer->transactions[0].pid = GRUB_USB_TRANSFER_TYPE_SETUP;
|
||||||
transfer->transactions[0].data = (char *) &setupdata;
|
transfer->transactions[0].data = setupdata_addr;
|
||||||
transfer->transactions[0].toggle = 0;
|
transfer->transactions[0].toggle = 0;
|
||||||
|
|
||||||
/* Now the data... XXX: Is this the right way to transfer control
|
/* Now the data... XXX: Is this the right way to transfer control
|
||||||
|
@ -99,13 +131,13 @@ grub_usb_control_msg (grub_usb_device_t dev,
|
||||||
tr->pid = GRUB_USB_TRANSFER_TYPE_IN;
|
tr->pid = GRUB_USB_TRANSFER_TYPE_IN;
|
||||||
else
|
else
|
||||||
tr->pid = GRUB_USB_TRANSFER_TYPE_OUT;
|
tr->pid = GRUB_USB_TRANSFER_TYPE_OUT;
|
||||||
tr->data = &data[i * max];
|
tr->data = data_addr + i * max;
|
||||||
size -= max;
|
size -= max;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* End with an empty OUT transaction. */
|
/* End with an empty OUT transaction. */
|
||||||
transfer->transactions[datablocks + 1].size = 0;
|
transfer->transactions[datablocks + 1].size = 0;
|
||||||
transfer->transactions[datablocks + 1].data = NULL;
|
transfer->transactions[datablocks + 1].data = 0;
|
||||||
if (reqtype & 128)
|
if (reqtype & 128)
|
||||||
transfer->transactions[datablocks + 1].pid = GRUB_USB_TRANSFER_TYPE_OUT;
|
transfer->transactions[datablocks + 1].pid = GRUB_USB_TRANSFER_TYPE_OUT;
|
||||||
else
|
else
|
||||||
|
@ -117,13 +149,17 @@ grub_usb_control_msg (grub_usb_device_t dev,
|
||||||
|
|
||||||
grub_free (transfer->transactions);
|
grub_free (transfer->transactions);
|
||||||
grub_free (transfer);
|
grub_free (transfer);
|
||||||
|
grub_dma_free (data_chunk);
|
||||||
|
grub_dma_free (setupdata_chunk);
|
||||||
|
|
||||||
|
grub_memcpy (data_in, data, size0);
|
||||||
|
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
static grub_usb_err_t
|
static grub_usb_err_t
|
||||||
grub_usb_bulk_readwrite (grub_usb_device_t dev,
|
grub_usb_bulk_readwrite (grub_usb_device_t dev,
|
||||||
int endpoint, grub_size_t size, char *data,
|
int endpoint, grub_size_t size0, char *data_in,
|
||||||
grub_transfer_type_t type)
|
grub_transfer_type_t type)
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
|
@ -132,6 +168,19 @@ grub_usb_bulk_readwrite (grub_usb_device_t dev,
|
||||||
unsigned int max;
|
unsigned int max;
|
||||||
grub_usb_err_t err;
|
grub_usb_err_t err;
|
||||||
int toggle = dev->toggle[endpoint];
|
int toggle = dev->toggle[endpoint];
|
||||||
|
char *data;
|
||||||
|
grub_uint32_t data_addr;
|
||||||
|
struct grub_pci_dma_chunk *data_chunk;
|
||||||
|
grub_size_t size = size0;
|
||||||
|
|
||||||
|
/* FIXME: avoid allocation any kind of buffer in a first place. */
|
||||||
|
data_chunk = grub_memalign_dma32 (128, size);
|
||||||
|
if (!data_chunk)
|
||||||
|
return GRUB_USB_ERR_INTERNAL;
|
||||||
|
data = grub_dma_get_virt (data_chunk);
|
||||||
|
data_addr = grub_dma_get_phys (data_chunk);
|
||||||
|
if (type == GRUB_USB_TRANSFER_TYPE_OUT)
|
||||||
|
grub_memcpy (data, data_in, size);
|
||||||
|
|
||||||
/* Use the maximum packet size given in the endpoint descriptor. */
|
/* Use the maximum packet size given in the endpoint descriptor. */
|
||||||
if (dev->initialized)
|
if (dev->initialized)
|
||||||
|
@ -150,7 +199,10 @@ grub_usb_bulk_readwrite (grub_usb_device_t dev,
|
||||||
/* Create a transfer. */
|
/* Create a transfer. */
|
||||||
transfer = grub_malloc (sizeof (struct grub_usb_transfer));
|
transfer = grub_malloc (sizeof (struct grub_usb_transfer));
|
||||||
if (! transfer)
|
if (! transfer)
|
||||||
return grub_errno;
|
{
|
||||||
|
grub_dma_free (data_chunk);
|
||||||
|
return grub_errno;
|
||||||
|
}
|
||||||
|
|
||||||
datablocks = ((size + max - 1) / max);
|
datablocks = ((size + max - 1) / max);
|
||||||
transfer->transcnt = datablocks;
|
transfer->transcnt = datablocks;
|
||||||
|
@ -167,6 +219,7 @@ grub_usb_bulk_readwrite (grub_usb_device_t dev,
|
||||||
if (! transfer->transactions)
|
if (! transfer->transactions)
|
||||||
{
|
{
|
||||||
grub_free (transfer);
|
grub_free (transfer);
|
||||||
|
grub_dma_free (data_chunk);
|
||||||
return grub_errno;
|
return grub_errno;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -181,7 +234,7 @@ grub_usb_bulk_readwrite (grub_usb_device_t dev,
|
||||||
tr->toggle = toggle;
|
tr->toggle = toggle;
|
||||||
toggle = toggle ? 0 : 1;
|
toggle = toggle ? 0 : 1;
|
||||||
tr->pid = type;
|
tr->pid = type;
|
||||||
tr->data = &data[i * max];
|
tr->data = data_addr + i * max;
|
||||||
size -= tr->size;
|
size -= tr->size;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -191,6 +244,10 @@ grub_usb_bulk_readwrite (grub_usb_device_t dev,
|
||||||
|
|
||||||
grub_free (transfer->transactions);
|
grub_free (transfer->transactions);
|
||||||
grub_free (transfer);
|
grub_free (transfer);
|
||||||
|
grub_dma_free (data_chunk);
|
||||||
|
|
||||||
|
if (type == GRUB_USB_TRANSFER_TYPE_IN)
|
||||||
|
grub_memcpy (data_in, data, size0);
|
||||||
|
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
|
@ -109,6 +109,15 @@ grub_pci_address_t EXPORT_FUNC(grub_pci_make_address) (grub_pci_device_t dev,
|
||||||
int reg);
|
int reg);
|
||||||
|
|
||||||
void EXPORT_FUNC(grub_pci_iterate) (grub_pci_iteratefunc_t hook);
|
void EXPORT_FUNC(grub_pci_iterate) (grub_pci_iteratefunc_t hook);
|
||||||
|
|
||||||
|
struct grub_pci_dma_chunk;
|
||||||
|
|
||||||
|
struct grub_pci_dma_chunk *EXPORT_FUNC(grub_memalign_dma32) (grub_size_t align,
|
||||||
|
grub_size_t size);
|
||||||
|
void EXPORT_FUNC(grub_dma_free) (struct grub_pci_dma_chunk *ch);
|
||||||
|
void *EXPORT_FUNC(grub_dma_get_virt) (struct grub_pci_dma_chunk *ch);
|
||||||
|
grub_uint32_t EXPORT_FUNC(grub_dma_get_phys) (struct grub_pci_dma_chunk *ch);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#endif /* GRUB_PCI_H */
|
#endif /* GRUB_PCI_H */
|
||||||
|
|
|
@ -37,7 +37,7 @@ struct grub_usb_transaction
|
||||||
int size;
|
int size;
|
||||||
int toggle;
|
int toggle;
|
||||||
grub_transfer_type_t pid;
|
grub_transfer_type_t pid;
|
||||||
char *data;
|
grub_uint32_t data;
|
||||||
};
|
};
|
||||||
typedef struct grub_usb_transaction *grub_usb_transaction_t;
|
typedef struct grub_usb_transaction *grub_usb_transaction_t;
|
||||||
|
|
||||||
|
|
|
@ -189,7 +189,7 @@ grub_real_dprintf (const char *file, const int line, const char *condition,
|
||||||
const char *debug = grub_env_get ("debug");
|
const char *debug = grub_env_get ("debug");
|
||||||
|
|
||||||
if (! debug)
|
if (! debug)
|
||||||
return;
|
debug = "usb,ohci";
|
||||||
|
|
||||||
if (grub_strword (debug, "all") || grub_strword (debug, condition))
|
if (grub_strword (debug, "all") || grub_strword (debug, condition))
|
||||||
{
|
{
|
||||||
|
@ -197,6 +197,7 @@ grub_real_dprintf (const char *file, const int line, const char *condition,
|
||||||
va_start (args, fmt);
|
va_start (args, fmt);
|
||||||
grub_vprintf (fmt, args);
|
grub_vprintf (fmt, args);
|
||||||
va_end (args);
|
va_end (args);
|
||||||
|
grub_refresh ();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue