459 lines
13 KiB
C
459 lines
13 KiB
C
|
/*
|
||
|
* Copyright (C) 2006 STMicroelectronics
|
||
|
*
|
||
|
* May be copied or modified under the terms of the GNU General Public
|
||
|
* License. See linux/COPYING for more information.
|
||
|
*
|
||
|
* Interfaces (where required) the co-processors on ST platforms based
|
||
|
* on multiprocessor architecture, for embedded products like Set-top-Box
|
||
|
* DVD, etc...
|
||
|
*
|
||
|
*/
|
||
|
|
||
|
#include <linux/kernel.h>
|
||
|
#include <linux/platform_device.h>
|
||
|
#include <linux/module.h>
|
||
|
#include <linux/bootmem.h>
|
||
|
#include <linux/fs.h>
|
||
|
#include <linux/init.h>
|
||
|
#include <linux/device.h>
|
||
|
#include <linux/firmware.h>
|
||
|
#include <linux/delay.h>
|
||
|
#include <linux/mm.h>
|
||
|
#include <linux/stm/coprocessor.h>
|
||
|
#include <asm/types.h>
|
||
|
#include <asm/sections.h>
|
||
|
#include <asm/uaccess.h>
|
||
|
#include <asm/io.h>
|
||
|
#include <asm/irq.h>
|
||
|
|
||
|
#ifdef CONFIG_PROC_FS
|
||
|
#include <linux/proc_fs.h>
|
||
|
#include <linux/seq_file.h>
|
||
|
|
||
|
#define COPR_FW_NAME_MAX 30
|
||
|
static int __init proc_st_coproc_init(void);
|
||
|
#endif
|
||
|
|
||
|
/* ---------------------------------------------------------------------------
|
||
|
* Local (declared out of order) functions
|
||
|
* ------------------------------------------------------------------------ */
|
||
|
|
||
|
static int __init parse_coproc_mem(char *from);
|
||
|
|
||
|
#undef dbg_print
|
||
|
|
||
|
#ifdef CONFIG_COPROCESSOR_DEBUG
|
||
|
#define dbg_print(fmt, args...) printk("%s: " fmt, __FUNCTION__ , ## args)
|
||
|
#else
|
||
|
#define dbg_print(fmt, args...)
|
||
|
#endif
|
||
|
|
||
|
/* ---------------------------------------------------------------------------
|
||
|
* Co-processor: Hardware dependent support
|
||
|
* This includes:
|
||
|
* - per platform device and memory addresses
|
||
|
* - platform dependent macros
|
||
|
* - HW dependent actions required by the generic APIs: Init,
|
||
|
* Open, Release, Ioctl (to reset, trigger the start (grant),
|
||
|
* peek and poke, etc...) functions
|
||
|
* ------------------------------------------------------------------------ */
|
||
|
|
||
|
extern struct coproc_board_info coproc_info;
|
||
|
|
||
|
/* ---------------------------------------------------------------------------
|
||
|
* Local data structure
|
||
|
* ------------------------------------------------------------------------ */
|
||
|
|
||
|
extern coproc_t coproc[]; /* The maximum number of coprocessors
|
||
|
* depends on platform type */
|
||
|
|
||
|
/* ---------------------------------------------------------------------------
|
||
|
* Co-processor driver APIs
|
||
|
* ---------------------------------------------------------------------------
|
||
|
*/
|
||
|
#define minor_2_firmware(min) ( (min) & 0x3f )
|
||
|
#define minor_2_device(min) ( ((min) >> 6) & 0x3)
|
||
|
|
||
|
static int st_coproc_open(struct inode *inode, struct file *file)
|
||
|
{
|
||
|
/*
|
||
|
** use minor number (ID) to access the current coproc. descriptor
|
||
|
*/
|
||
|
char firm_file[COPR_FW_NAME_MAX];
|
||
|
char number[10];
|
||
|
unsigned long minor = MINOR((file)->f_dentry->d_inode->i_rdev);
|
||
|
unsigned long id_device = minor_2_device(minor);
|
||
|
unsigned long id_firmware = minor_2_firmware(minor);
|
||
|
const struct firmware *fw = NULL;
|
||
|
int res;
|
||
|
|
||
|
coproc_t *cop = &coproc[id_device];
|
||
|
|
||
|
if (cop->control & COPROC_IN_USE)
|
||
|
return -EBUSY;
|
||
|
/* Now call the platform dependent open stage */
|
||
|
coproc_cpu_open(cop);
|
||
|
/* Build the firmware file name.
|
||
|
* We use the standard name: "st_firmware_XX_XX.elf"
|
||
|
* to specify the device number and the firmware number
|
||
|
*/
|
||
|
|
||
|
strcpy(firm_file, "st_firmware_");
|
||
|
sprintf(number, "%ld", id_device);
|
||
|
strcat(firm_file, number);
|
||
|
strcat(firm_file, "_");
|
||
|
sprintf(number, "%ld", id_firmware);
|
||
|
strcat(firm_file, number);
|
||
|
strcat(firm_file, ".elf");
|
||
|
|
||
|
dbg_print("Asking the file %s for %s.%u\n", firm_file, cop->pdev.name,cop->pdev.id);
|
||
|
if (request_firmware(&fw, firm_file, &(cop->pdev.dev)) == 0) {
|
||
|
unsigned long boot_address;
|
||
|
|
||
|
cop->control |= COPROC_IN_USE;
|
||
|
/* move the firmware in the coprocessor memory */
|
||
|
dbg_print("Received firmware size %d bytes\n", fw->size - 4);
|
||
|
dbg_print("cop->ram_size = 0x%x\n", cop->ram_size);
|
||
|
dbg_print("cop->ram_offset = 0x%x\n", (unsigned int)cop->ram_offset);
|
||
|
/*
|
||
|
* The last 4 bytes in the fw->data buffer
|
||
|
* aren't code.
|
||
|
* They are the boot vma (relocated) address!
|
||
|
*/
|
||
|
memcpy(&boot_address, (fw->data) + (fw->size - 4), 4);
|
||
|
dbg_print("boot address = 0x%x\n", (unsigned int)boot_address);
|
||
|
memcpy((int*)cop->vma_address, fw->data, fw->size - 4);
|
||
|
release_firmware(fw);
|
||
|
dbg_print("Run the Firmware code\n");
|
||
|
coproc_cpu_grant(cop, (unsigned int)boot_address); //7100 only...
|
||
|
res = 0;
|
||
|
} else {
|
||
|
dbg_print("Error on Firmware Download\n");
|
||
|
res = -EINVAL;
|
||
|
}
|
||
|
return res;
|
||
|
}
|
||
|
|
||
|
static int st_coproc_release(struct inode *inode, struct file *file)
|
||
|
{
|
||
|
coproc_t *cop = FILE_2_COP(coproc, file);
|
||
|
|
||
|
coproc_cpu_release(cop);
|
||
|
cop->control &= ~COPROC_IN_USE;
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static struct file_operations coproc_fops = {
|
||
|
open:st_coproc_open,
|
||
|
release:st_coproc_release
|
||
|
};
|
||
|
|
||
|
/* Start: ST-Coprocessor Device Attribute on SysFs*/
|
||
|
static ssize_t st_copro_show_running(struct device *dev, struct device_attribute *attr, char *buf)
|
||
|
{
|
||
|
coproc_t *cop = container_of(dev, coproc_t, pdev.dev);
|
||
|
return sprintf(buf, "%d", cop->control & COPROC_IN_USE);
|
||
|
}
|
||
|
|
||
|
static DEVICE_ATTR(running, S_IRUGO, st_copro_show_running, NULL);
|
||
|
|
||
|
static ssize_t st_copro_show_mem_size(struct device *dev, struct device_attribute *attr, char *buf)
|
||
|
{
|
||
|
coproc_t *cop = container_of(dev, coproc_t, pdev.dev);
|
||
|
return sprintf(buf, "0x%x", cop->ram_size);
|
||
|
}
|
||
|
|
||
|
static DEVICE_ATTR(mem_size, S_IRUGO, st_copro_show_mem_size, NULL);
|
||
|
|
||
|
static ssize_t st_copro_show_mem_base(struct device *dev, struct device_attribute *attr, char *buf)
|
||
|
{
|
||
|
coproc_t *cop = container_of(dev, coproc_t, pdev.dev);
|
||
|
return sprintf(buf, "0x%x", (int)COPR_ADDR(cop, 0));
|
||
|
}
|
||
|
|
||
|
static DEVICE_ATTR(mem_base, S_IRUGO, st_copro_show_mem_base, NULL);
|
||
|
/* End: ST-Coprocessor Device Attribute SysFs*/
|
||
|
|
||
|
static int st_coproc_driver_probe(struct platform_device *dev)
|
||
|
{
|
||
|
if (!strncmp("st2", dev->name, 3))
|
||
|
return 1;
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
#if defined(CONFIG_PM)
|
||
|
static int st_coproc_suspend(struct platform_device * dev, pm_message_t state)
|
||
|
{
|
||
|
printk("st_coproc_suspend: %s.%u down\n",dev->name,dev->id);
|
||
|
/*
|
||
|
*
|
||
|
*/
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static int st_coproc_resume(struct platform_device * dev)
|
||
|
{
|
||
|
printk("st_coproc_resume: %s.%u up\n",dev->name,dev->id);
|
||
|
/*
|
||
|
*
|
||
|
*/
|
||
|
return 0;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
static struct platform_driver st_coproc_driver = {
|
||
|
.driver.name = "st-copro",
|
||
|
.driver.owner = THIS_MODULE,
|
||
|
.probe = st_coproc_driver_probe,
|
||
|
#if defined(CONFIG_PM)
|
||
|
.suspend =st_coproc_suspend,
|
||
|
.resume = st_coproc_resume,
|
||
|
#endif
|
||
|
};
|
||
|
|
||
|
static struct class *coproc_dev_class;
|
||
|
static int __init st_coproc_init(void)
|
||
|
{
|
||
|
int i;
|
||
|
int frmw_idx;
|
||
|
coproc_t *cop;
|
||
|
struct platform_device *pdev;
|
||
|
|
||
|
printk("STMicroelectronics - Coprocessors %s Init\n", coproc_info.name);
|
||
|
|
||
|
if (platform_driver_register(&st_coproc_driver)) {
|
||
|
printk(KERN_ERR
|
||
|
"Error on ST-Coprocessor device driver registration\n");
|
||
|
return (-EAGAIN);
|
||
|
}
|
||
|
|
||
|
if (register_chrdev(COPROCESSOR_MAJOR, coproc_info.name, &coproc_fops)) {
|
||
|
printk("Can't allocate major %d for ST Coprocessor Devices\n",
|
||
|
COPROCESSOR_MAJOR);
|
||
|
platform_driver_unregister(&st_coproc_driver);
|
||
|
return (-EAGAIN);
|
||
|
}
|
||
|
|
||
|
coproc_dev_class = class_create(THIS_MODULE, "coproc-dev");
|
||
|
|
||
|
for (cop = &coproc[0], i = 0; i < coproc_info.max_coprs; i++, cop++) {
|
||
|
/**
|
||
|
** Nodes:
|
||
|
** STb7100 : /dev/st231-0 c 63 0
|
||
|
** /dev/st231-1 c 63 1
|
||
|
**/
|
||
|
if (!cop->ram_offset) {
|
||
|
printk("st-coprocessor-%d: No RAM reserved\n", i);
|
||
|
cop->control &= ~COPROC_SPACE_ALLOCATE;
|
||
|
} else {
|
||
|
cop->control |= COPROC_SPACE_ALLOCATE;
|
||
|
cop->vma_address =
|
||
|
(int)ioremap_nocache(cop->ram_offset, cop->ram_size);
|
||
|
}
|
||
|
/*
|
||
|
* Setup and Add the device entries in the SysFS
|
||
|
*/
|
||
|
pdev = &(cop->pdev);
|
||
|
memset(pdev, 0, sizeof(struct platform_device));
|
||
|
pdev->name = coproc_info.name;
|
||
|
pdev->id = i;
|
||
|
pdev->dev.driver = &st_coproc_driver.driver;
|
||
|
/* Now complete with the platform dependent init stage */
|
||
|
if (coproc_cpu_init(cop)){
|
||
|
printk(KERN_ERR
|
||
|
"CPU %d : HW dep. initialization failed!\n", i);
|
||
|
return (1);
|
||
|
}
|
||
|
if (platform_device_register(pdev))
|
||
|
printk(KERN_ERR
|
||
|
"Error on ST-Coprocessor device registration\n");
|
||
|
else {
|
||
|
/* Add the attributes on the device */
|
||
|
if(device_create_file(&pdev->dev, &dev_attr_mem_base) |
|
||
|
device_create_file(&pdev->dev, &dev_attr_mem_size) |
|
||
|
device_create_file(&pdev->dev, &dev_attr_running))
|
||
|
printk(KERN_ERR "Error to add attribute to the coprocessor device\n");
|
||
|
/*
|
||
|
* Create the six device file [firmware]
|
||
|
* for each coprocessor via Discovery System
|
||
|
*/
|
||
|
for(frmw_idx=0; frmw_idx < 10; ++frmw_idx)
|
||
|
/* Be carefull the '6' used in MKDEV(..) depends on
|
||
|
* minor number device file translation */
|
||
|
cop->dev = device_create(coproc_dev_class, NULL,
|
||
|
MKDEV(COPROCESSOR_MAJOR,pdev->id<<6 | frmw_idx),
|
||
|
NULL,"st231-%d-%d", pdev->id, frmw_idx);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#ifdef CONFIG_PROC_FS
|
||
|
proc_st_coproc_init();
|
||
|
#endif
|
||
|
|
||
|
return (0);
|
||
|
}
|
||
|
|
||
|
static void __exit st_coproc_exit(void)
|
||
|
{
|
||
|
dbg_print("Release coprocessor module...\n");
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* Parse the optional kernel argument:
|
||
|
*
|
||
|
* ... coprocessor_mem=size_0@phis_address_0, size_1@phis_address_1
|
||
|
*
|
||
|
* It seems to be reasonable to assume that in a "staically partitioned
|
||
|
* RAM layout", the regions of RAM assigned to each slave processor are
|
||
|
* not scattered in memory!
|
||
|
*/
|
||
|
static int __init parse_coproc_mem(char *from)
|
||
|
{
|
||
|
char *cmdl = (from); /* start scan from '=' char */
|
||
|
u_long size, addr;
|
||
|
int i = 0;
|
||
|
char *error_msg;
|
||
|
static char size_error[] __initdata =
|
||
|
KERN_ERR "st-coprocessor: Error parsing size\n";
|
||
|
static char addr_error[] __initdata =
|
||
|
KERN_ERR "st-coprocessor: Error parsing address\n";
|
||
|
static char too_many_warn[] __initdata =
|
||
|
KERN_WARNING "st-coprocessor: More regions than coprocessors\n";
|
||
|
static char alloc_error[] __initdata =
|
||
|
KERN_ERR "st-coprocessor: Failed to reserve memoryat 0x%08x\n";
|
||
|
|
||
|
while (*cmdl && (i < coproc_info.max_coprs)) {
|
||
|
size = memparse(cmdl, &cmdl);
|
||
|
if (*cmdl != '@') {
|
||
|
error_msg = size_error;
|
||
|
goto args_error;
|
||
|
}
|
||
|
addr = memparse(cmdl+1, &cmdl);
|
||
|
if (*cmdl) {
|
||
|
if (*cmdl++ != ',') {
|
||
|
error_msg = addr_error;
|
||
|
goto args_error;
|
||
|
}
|
||
|
}
|
||
|
coproc[i].ram_offset = addr;
|
||
|
coproc[i].ram_size = size;
|
||
|
++i;
|
||
|
}
|
||
|
|
||
|
if (*cmdl) {
|
||
|
printk(too_many_warn);
|
||
|
}
|
||
|
|
||
|
for (i = 0; i < coproc_info.max_coprs; ++i) {
|
||
|
if (coproc[i].ram_size) {
|
||
|
void* mem;
|
||
|
addr = coproc[i].ram_offset;
|
||
|
size = coproc[i].ram_size;
|
||
|
mem = __alloc_bootmem_nopanic(size, PAGE_SIZE, addr);
|
||
|
if (mem != __va(addr)) {
|
||
|
if (mem) {
|
||
|
free_bootmem(virt_to_phys(mem), size);
|
||
|
}
|
||
|
/* At this point, if addr overlaps kernel
|
||
|
* memory, coprocessor won't be allocated.
|
||
|
*/
|
||
|
if (coproc_check_area(addr, size, i, coproc))
|
||
|
printk(alloc_error, addr);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return 1;
|
||
|
|
||
|
args_error:
|
||
|
printk(error_msg);
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
__setup("coprocessor_mem=", parse_coproc_mem);
|
||
|
|
||
|
MODULE_DESCRIPTION("Co-processor manager for multi-core devices");
|
||
|
MODULE_AUTHOR("STMicroelectronics Limited");
|
||
|
MODULE_VERSION("0.3");
|
||
|
MODULE_LICENSE("GPL");
|
||
|
|
||
|
module_init(st_coproc_init);
|
||
|
module_exit(st_coproc_exit);
|
||
|
|
||
|
#ifdef CONFIG_PROC_FS
|
||
|
|
||
|
static int show_st_coproc(struct seq_file *m, void *v)
|
||
|
{
|
||
|
int i;
|
||
|
coproc_t *cop;
|
||
|
|
||
|
seq_printf(m, "Coprocessors: %d %s\n", coproc_info.max_coprs,
|
||
|
coproc_info.name);
|
||
|
seq_printf(m,
|
||
|
" CPU (dev) Host addr. Copr. addr. Size\n");
|
||
|
seq_printf(m,
|
||
|
" ----------------------------------------------------------- --------\n");
|
||
|
for (i = 0, cop = &coproc[0]; i < coproc_info.max_coprs; i++, cop++) {
|
||
|
seq_printf(m, " /dev/%s-%u ", cop->pdev.name,cop->pdev.id);
|
||
|
if (cop->ram_size == 0)
|
||
|
seq_printf(m, "not allocated!\n");
|
||
|
else
|
||
|
seq_printf(m,
|
||
|
"0x%08lx 0x%08lx 0x%08x (%2d Mb)\n",
|
||
|
(HOST_ADDR(cop, 0)), COPR_ADDR(cop, 0),
|
||
|
cop->ram_size, (cop->ram_size / MEGA));
|
||
|
}
|
||
|
seq_printf(m, "\n");
|
||
|
|
||
|
coproc_proc_other_info(cop, m);
|
||
|
return (0);
|
||
|
}
|
||
|
|
||
|
static void *st_coproc_seq_start(struct seq_file *m, loff_t * pos)
|
||
|
{
|
||
|
return (void *)(*pos == 0);
|
||
|
}
|
||
|
|
||
|
static void *st_coproc_seq_next(struct seq_file *m, void *v, loff_t * pos)
|
||
|
{
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
static void st_coproc_seq_stop(struct seq_file *m, void *v)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
static struct seq_operations proc_st_coproc_op = {
|
||
|
start:st_coproc_seq_start,
|
||
|
next:st_coproc_seq_next,
|
||
|
stop:st_coproc_seq_stop,
|
||
|
show:show_st_coproc,
|
||
|
};
|
||
|
|
||
|
static int proc_st_coproc_open(struct inode *inode, struct file *file)
|
||
|
{
|
||
|
return seq_open(file, &proc_st_coproc_op);
|
||
|
}
|
||
|
|
||
|
static struct file_operations proc_st_coproc_operations = {
|
||
|
open:proc_st_coproc_open,
|
||
|
read:seq_read,
|
||
|
llseek:seq_lseek,
|
||
|
release:seq_release,
|
||
|
};
|
||
|
|
||
|
static int __init proc_st_coproc_init(void)
|
||
|
{
|
||
|
struct proc_dir_entry *entry;
|
||
|
entry = create_proc_entry("coprocessor", 0, NULL);
|
||
|
if (entry != NULL) {
|
||
|
entry->proc_fops = &proc_st_coproc_operations;
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
#endif /* CONFIG_PROC_FS */
|