mirror of
https://github.com/AsahiLinux/u-boot
synced 2025-01-07 10:48:54 +00:00
83d290c56f
When U-Boot started using SPDX tags we were among the early adopters and there weren't a lot of other examples to borrow from. So we picked the area of the file that usually had a full license text and replaced it with an appropriate SPDX-License-Identifier: entry. Since then, the Linux Kernel has adopted SPDX tags and they place it as the very first line in a file (except where shebangs are used, then it's second line) and with slightly different comment styles than us. In part due to community overlap, in part due to better tag visibility and in part for other minor reasons, switch over to that style. This commit changes all instances where we have a single declared license in the tag as both the before and after are identical in tag contents. There's also a few places where I found we did not have a tag and have introduced one. Signed-off-by: Tom Rini <trini@konsulko.com>
285 lines
6 KiB
C
285 lines
6 KiB
C
// SPDX-License-Identifier: GPL-2.0+
|
|
/*
|
|
* PCI emulation device which swaps the case of text
|
|
*
|
|
* Copyright (c) 2014 Google, Inc
|
|
* Written by Simon Glass <sjg@chromium.org>
|
|
*/
|
|
|
|
#include <common.h>
|
|
#include <dm.h>
|
|
#include <errno.h>
|
|
#include <pci.h>
|
|
#include <asm/test.h>
|
|
#include <linux/ctype.h>
|
|
|
|
/**
|
|
* struct swap_case_platdata - platform data for this device
|
|
*
|
|
* @command: Current PCI command value
|
|
* @bar: Current base address values
|
|
*/
|
|
struct swap_case_platdata {
|
|
u16 command;
|
|
u32 bar[2];
|
|
};
|
|
|
|
#define offset_to_barnum(offset) \
|
|
(((offset) - PCI_BASE_ADDRESS_0) / sizeof(u32))
|
|
|
|
enum {
|
|
MEM_TEXT_SIZE = 0x100,
|
|
};
|
|
|
|
enum swap_case_op {
|
|
OP_TO_LOWER,
|
|
OP_TO_UPPER,
|
|
OP_SWAP,
|
|
};
|
|
|
|
static struct pci_bar {
|
|
int type;
|
|
u32 size;
|
|
} barinfo[] = {
|
|
{ PCI_BASE_ADDRESS_SPACE_IO, 1 },
|
|
{ PCI_BASE_ADDRESS_MEM_TYPE_32, MEM_TEXT_SIZE },
|
|
{ 0, 0 },
|
|
{ 0, 0 },
|
|
{ 0, 0 },
|
|
{ 0, 0 },
|
|
};
|
|
|
|
struct swap_case_priv {
|
|
enum swap_case_op op;
|
|
char mem_text[MEM_TEXT_SIZE];
|
|
};
|
|
|
|
static int sandbox_swap_case_get_devfn(struct udevice *dev)
|
|
{
|
|
struct pci_child_platdata *plat = dev_get_parent_platdata(dev);
|
|
|
|
return plat->devfn;
|
|
}
|
|
|
|
static int sandbox_swap_case_read_config(struct udevice *emul, uint offset,
|
|
ulong *valuep, enum pci_size_t size)
|
|
{
|
|
struct swap_case_platdata *plat = dev_get_platdata(emul);
|
|
|
|
switch (offset) {
|
|
case PCI_COMMAND:
|
|
*valuep = plat->command;
|
|
break;
|
|
case PCI_HEADER_TYPE:
|
|
*valuep = 0;
|
|
break;
|
|
case PCI_VENDOR_ID:
|
|
*valuep = SANDBOX_PCI_VENDOR_ID;
|
|
break;
|
|
case PCI_DEVICE_ID:
|
|
*valuep = SANDBOX_PCI_DEVICE_ID;
|
|
break;
|
|
case PCI_CLASS_DEVICE:
|
|
if (size == PCI_SIZE_8) {
|
|
*valuep = SANDBOX_PCI_CLASS_SUB_CODE;
|
|
} else {
|
|
*valuep = (SANDBOX_PCI_CLASS_CODE << 8) |
|
|
SANDBOX_PCI_CLASS_SUB_CODE;
|
|
}
|
|
break;
|
|
case PCI_CLASS_CODE:
|
|
*valuep = SANDBOX_PCI_CLASS_CODE;
|
|
break;
|
|
case PCI_BASE_ADDRESS_0:
|
|
case PCI_BASE_ADDRESS_1:
|
|
case PCI_BASE_ADDRESS_2:
|
|
case PCI_BASE_ADDRESS_3:
|
|
case PCI_BASE_ADDRESS_4:
|
|
case PCI_BASE_ADDRESS_5: {
|
|
int barnum;
|
|
u32 *bar, result;
|
|
|
|
barnum = offset_to_barnum(offset);
|
|
bar = &plat->bar[barnum];
|
|
|
|
result = *bar;
|
|
if (*bar == 0xffffffff) {
|
|
if (barinfo[barnum].type) {
|
|
result = (~(barinfo[barnum].size - 1) &
|
|
PCI_BASE_ADDRESS_IO_MASK) |
|
|
PCI_BASE_ADDRESS_SPACE_IO;
|
|
} else {
|
|
result = (~(barinfo[barnum].size - 1) &
|
|
PCI_BASE_ADDRESS_MEM_MASK) |
|
|
PCI_BASE_ADDRESS_MEM_TYPE_32;
|
|
}
|
|
}
|
|
debug("r bar %d=%x\n", barnum, result);
|
|
*valuep = result;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int sandbox_swap_case_write_config(struct udevice *emul, uint offset,
|
|
ulong value, enum pci_size_t size)
|
|
{
|
|
struct swap_case_platdata *plat = dev_get_platdata(emul);
|
|
|
|
switch (offset) {
|
|
case PCI_COMMAND:
|
|
plat->command = value;
|
|
break;
|
|
case PCI_BASE_ADDRESS_0:
|
|
case PCI_BASE_ADDRESS_1: {
|
|
int barnum;
|
|
u32 *bar;
|
|
|
|
barnum = offset_to_barnum(offset);
|
|
bar = &plat->bar[barnum];
|
|
|
|
debug("w bar %d=%lx\n", barnum, value);
|
|
*bar = value;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int sandbox_swap_case_find_bar(struct udevice *emul, unsigned int addr,
|
|
int *barnump, unsigned int *offsetp)
|
|
{
|
|
struct swap_case_platdata *plat = dev_get_platdata(emul);
|
|
int barnum;
|
|
|
|
for (barnum = 0; barnum < ARRAY_SIZE(barinfo); barnum++) {
|
|
unsigned int size = barinfo[barnum].size;
|
|
|
|
if (addr >= plat->bar[barnum] &&
|
|
addr < plat->bar[barnum] + size) {
|
|
*barnump = barnum;
|
|
*offsetp = addr - plat->bar[barnum];
|
|
return 0;
|
|
}
|
|
}
|
|
*barnump = -1;
|
|
|
|
return -ENOENT;
|
|
}
|
|
|
|
static void sandbox_swap_case_do_op(enum swap_case_op op, char *str, int len)
|
|
{
|
|
for (; len > 0; len--, str++) {
|
|
switch (op) {
|
|
case OP_TO_UPPER:
|
|
*str = toupper(*str);
|
|
break;
|
|
case OP_TO_LOWER:
|
|
*str = tolower(*str);
|
|
break;
|
|
case OP_SWAP:
|
|
if (isupper(*str))
|
|
*str = tolower(*str);
|
|
else
|
|
*str = toupper(*str);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
int sandbox_swap_case_read_io(struct udevice *dev, unsigned int addr,
|
|
ulong *valuep, enum pci_size_t size)
|
|
{
|
|
struct swap_case_priv *priv = dev_get_priv(dev);
|
|
unsigned int offset;
|
|
int barnum;
|
|
int ret;
|
|
|
|
ret = sandbox_swap_case_find_bar(dev, addr, &barnum, &offset);
|
|
if (ret)
|
|
return ret;
|
|
|
|
if (barnum == 0 && offset == 0)
|
|
*valuep = (*valuep & ~0xff) | priv->op;
|
|
|
|
return 0;
|
|
}
|
|
|
|
int sandbox_swap_case_write_io(struct udevice *dev, unsigned int addr,
|
|
ulong value, enum pci_size_t size)
|
|
{
|
|
struct swap_case_priv *priv = dev_get_priv(dev);
|
|
unsigned int offset;
|
|
int barnum;
|
|
int ret;
|
|
|
|
ret = sandbox_swap_case_find_bar(dev, addr, &barnum, &offset);
|
|
if (ret)
|
|
return ret;
|
|
if (barnum == 0 && offset == 0)
|
|
priv->op = value;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int sandbox_swap_case_map_physmem(struct udevice *dev,
|
|
phys_addr_t addr, unsigned long *lenp, void **ptrp)
|
|
{
|
|
struct swap_case_priv *priv = dev_get_priv(dev);
|
|
unsigned int offset, avail;
|
|
int barnum;
|
|
int ret;
|
|
|
|
ret = sandbox_swap_case_find_bar(dev, addr, &barnum, &offset);
|
|
if (ret)
|
|
return ret;
|
|
if (barnum == 1) {
|
|
*ptrp = priv->mem_text + offset;
|
|
avail = barinfo[1].size - offset;
|
|
if (avail > barinfo[1].size)
|
|
*lenp = 0;
|
|
else
|
|
*lenp = min(*lenp, (ulong)avail);
|
|
|
|
return 0;
|
|
}
|
|
|
|
return -ENOENT;
|
|
}
|
|
|
|
static int sandbox_swap_case_unmap_physmem(struct udevice *dev,
|
|
const void *vaddr, unsigned long len)
|
|
{
|
|
struct swap_case_priv *priv = dev_get_priv(dev);
|
|
|
|
sandbox_swap_case_do_op(priv->op, (void *)vaddr, len);
|
|
|
|
return 0;
|
|
}
|
|
|
|
struct dm_pci_emul_ops sandbox_swap_case_emul_ops = {
|
|
.get_devfn = sandbox_swap_case_get_devfn,
|
|
.read_config = sandbox_swap_case_read_config,
|
|
.write_config = sandbox_swap_case_write_config,
|
|
.read_io = sandbox_swap_case_read_io,
|
|
.write_io = sandbox_swap_case_write_io,
|
|
.map_physmem = sandbox_swap_case_map_physmem,
|
|
.unmap_physmem = sandbox_swap_case_unmap_physmem,
|
|
};
|
|
|
|
static const struct udevice_id sandbox_swap_case_ids[] = {
|
|
{ .compatible = "sandbox,swap-case" },
|
|
{ }
|
|
};
|
|
|
|
U_BOOT_DRIVER(sandbox_swap_case_emul) = {
|
|
.name = "sandbox_swap_case_emul",
|
|
.id = UCLASS_PCI_EMUL,
|
|
.of_match = sandbox_swap_case_ids,
|
|
.ops = &sandbox_swap_case_emul_ops,
|
|
.priv_auto_alloc_size = sizeof(struct swap_case_priv),
|
|
.platdata_auto_alloc_size = sizeof(struct swap_case_platdata),
|
|
};
|