7240f1f183
Patch from Lennert Buytenhek Make the uengine loader use ixp2000_reg_wrb in the right places. Signed-off-by: Lennert Buytenhek <buytenh@wantstofly.org> Signed-off-by: Russell King <rmk+kernel@arm.linux.org.uk>
473 lines
11 KiB
C
473 lines
11 KiB
C
/*
|
|
* Generic library functions for the microengines found on the Intel
|
|
* IXP2000 series of network processors.
|
|
*
|
|
* Copyright (C) 2004, 2005 Lennert Buytenhek <buytenh@wantstofly.org>
|
|
* Dedicated to Marija Kulikova.
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU Lesser General Public License as
|
|
* published by the Free Software Foundation; either version 2.1 of the
|
|
* License, or (at your option) any later version.
|
|
*/
|
|
|
|
#include <linux/config.h>
|
|
#include <linux/kernel.h>
|
|
#include <linux/init.h>
|
|
#include <linux/slab.h>
|
|
#include <linux/module.h>
|
|
#include <linux/string.h>
|
|
#include <asm/hardware.h>
|
|
#include <asm/arch/ixp2000-regs.h>
|
|
#include <asm/arch/uengine.h>
|
|
#include <asm/io.h>
|
|
|
|
#define USTORE_ADDRESS 0x000
|
|
#define USTORE_DATA_LOWER 0x004
|
|
#define USTORE_DATA_UPPER 0x008
|
|
#define CTX_ENABLES 0x018
|
|
#define CC_ENABLE 0x01c
|
|
#define CSR_CTX_POINTER 0x020
|
|
#define INDIRECT_CTX_STS 0x040
|
|
#define ACTIVE_CTX_STS 0x044
|
|
#define INDIRECT_CTX_SIG_EVENTS 0x048
|
|
#define INDIRECT_CTX_WAKEUP_EVENTS 0x050
|
|
#define NN_PUT 0x080
|
|
#define NN_GET 0x084
|
|
#define TIMESTAMP_LOW 0x0c0
|
|
#define TIMESTAMP_HIGH 0x0c4
|
|
#define T_INDEX_BYTE_INDEX 0x0f4
|
|
#define LOCAL_CSR_STATUS 0x180
|
|
|
|
u32 ixp2000_uengine_mask;
|
|
|
|
static void *ixp2000_uengine_csr_area(int uengine)
|
|
{
|
|
return ((void *)IXP2000_UENGINE_CSR_VIRT_BASE) + (uengine << 10);
|
|
}
|
|
|
|
/*
|
|
* LOCAL_CSR_STATUS=1 after a read or write to a microengine's CSR
|
|
* space means that the microengine we tried to access was also trying
|
|
* to access its own CSR space on the same clock cycle as we did. When
|
|
* this happens, we lose the arbitration process by default, and the
|
|
* read or write we tried to do was not actually performed, so we try
|
|
* again until it succeeds.
|
|
*/
|
|
u32 ixp2000_uengine_csr_read(int uengine, int offset)
|
|
{
|
|
void *uebase;
|
|
u32 *local_csr_status;
|
|
u32 *reg;
|
|
u32 value;
|
|
|
|
uebase = ixp2000_uengine_csr_area(uengine);
|
|
|
|
local_csr_status = (u32 *)(uebase + LOCAL_CSR_STATUS);
|
|
reg = (u32 *)(uebase + offset);
|
|
do {
|
|
value = ixp2000_reg_read(reg);
|
|
} while (ixp2000_reg_read(local_csr_status) & 1);
|
|
|
|
return value;
|
|
}
|
|
EXPORT_SYMBOL(ixp2000_uengine_csr_read);
|
|
|
|
void ixp2000_uengine_csr_write(int uengine, int offset, u32 value)
|
|
{
|
|
void *uebase;
|
|
u32 *local_csr_status;
|
|
u32 *reg;
|
|
|
|
uebase = ixp2000_uengine_csr_area(uengine);
|
|
|
|
local_csr_status = (u32 *)(uebase + LOCAL_CSR_STATUS);
|
|
reg = (u32 *)(uebase + offset);
|
|
do {
|
|
ixp2000_reg_write(reg, value);
|
|
} while (ixp2000_reg_read(local_csr_status) & 1);
|
|
}
|
|
EXPORT_SYMBOL(ixp2000_uengine_csr_write);
|
|
|
|
void ixp2000_uengine_reset(u32 uengine_mask)
|
|
{
|
|
ixp2000_reg_wrb(IXP2000_RESET1, uengine_mask & ixp2000_uengine_mask);
|
|
ixp2000_reg_wrb(IXP2000_RESET1, 0);
|
|
}
|
|
EXPORT_SYMBOL(ixp2000_uengine_reset);
|
|
|
|
void ixp2000_uengine_set_mode(int uengine, u32 mode)
|
|
{
|
|
/*
|
|
* CTL_STR_PAR_EN: unconditionally enable parity checking on
|
|
* control store.
|
|
*/
|
|
mode |= 0x10000000;
|
|
ixp2000_uengine_csr_write(uengine, CTX_ENABLES, mode);
|
|
|
|
/*
|
|
* Enable updating of condition codes.
|
|
*/
|
|
ixp2000_uengine_csr_write(uengine, CC_ENABLE, 0x00002000);
|
|
|
|
/*
|
|
* Initialise other per-microengine registers.
|
|
*/
|
|
ixp2000_uengine_csr_write(uengine, NN_PUT, 0x00);
|
|
ixp2000_uengine_csr_write(uengine, NN_GET, 0x00);
|
|
ixp2000_uengine_csr_write(uengine, T_INDEX_BYTE_INDEX, 0);
|
|
}
|
|
EXPORT_SYMBOL(ixp2000_uengine_set_mode);
|
|
|
|
static int make_even_parity(u32 x)
|
|
{
|
|
return hweight32(x) & 1;
|
|
}
|
|
|
|
static void ustore_write(int uengine, u64 insn)
|
|
{
|
|
/*
|
|
* Generate even parity for top and bottom 20 bits.
|
|
*/
|
|
insn |= (u64)make_even_parity((insn >> 20) & 0x000fffff) << 41;
|
|
insn |= (u64)make_even_parity(insn & 0x000fffff) << 40;
|
|
|
|
/*
|
|
* Write to microstore. The second write auto-increments
|
|
* the USTORE_ADDRESS index register.
|
|
*/
|
|
ixp2000_uengine_csr_write(uengine, USTORE_DATA_LOWER, (u32)insn);
|
|
ixp2000_uengine_csr_write(uengine, USTORE_DATA_UPPER, (u32)(insn >> 32));
|
|
}
|
|
|
|
void ixp2000_uengine_load_microcode(int uengine, u8 *ucode, int insns)
|
|
{
|
|
int i;
|
|
|
|
/*
|
|
* Start writing to microstore at address 0.
|
|
*/
|
|
ixp2000_uengine_csr_write(uengine, USTORE_ADDRESS, 0x80000000);
|
|
for (i = 0; i < insns; i++) {
|
|
u64 insn;
|
|
|
|
insn = (((u64)ucode[0]) << 32) |
|
|
(((u64)ucode[1]) << 24) |
|
|
(((u64)ucode[2]) << 16) |
|
|
(((u64)ucode[3]) << 8) |
|
|
((u64)ucode[4]);
|
|
ucode += 5;
|
|
|
|
ustore_write(uengine, insn);
|
|
}
|
|
|
|
/*
|
|
* Pad with a few NOPs at the end (to avoid the microengine
|
|
* aborting as it prefetches beyond the last instruction), unless
|
|
* we run off the end of the instruction store first, at which
|
|
* point the address register will wrap back to zero.
|
|
*/
|
|
for (i = 0; i < 4; i++) {
|
|
u32 addr;
|
|
|
|
addr = ixp2000_uengine_csr_read(uengine, USTORE_ADDRESS);
|
|
if (addr == 0x80000000)
|
|
break;
|
|
ustore_write(uengine, 0xf0000c0300ULL);
|
|
}
|
|
|
|
/*
|
|
* End programming.
|
|
*/
|
|
ixp2000_uengine_csr_write(uengine, USTORE_ADDRESS, 0x00000000);
|
|
}
|
|
EXPORT_SYMBOL(ixp2000_uengine_load_microcode);
|
|
|
|
void ixp2000_uengine_init_context(int uengine, int context, int pc)
|
|
{
|
|
/*
|
|
* Select the right context for indirect access.
|
|
*/
|
|
ixp2000_uengine_csr_write(uengine, CSR_CTX_POINTER, context);
|
|
|
|
/*
|
|
* Initialise signal masks to immediately go to Ready state.
|
|
*/
|
|
ixp2000_uengine_csr_write(uengine, INDIRECT_CTX_SIG_EVENTS, 1);
|
|
ixp2000_uengine_csr_write(uengine, INDIRECT_CTX_WAKEUP_EVENTS, 1);
|
|
|
|
/*
|
|
* Set program counter.
|
|
*/
|
|
ixp2000_uengine_csr_write(uengine, INDIRECT_CTX_STS, pc);
|
|
}
|
|
EXPORT_SYMBOL(ixp2000_uengine_init_context);
|
|
|
|
void ixp2000_uengine_start_contexts(int uengine, u8 ctx_mask)
|
|
{
|
|
u32 mask;
|
|
|
|
/*
|
|
* Enable the specified context to go to Executing state.
|
|
*/
|
|
mask = ixp2000_uengine_csr_read(uengine, CTX_ENABLES);
|
|
mask |= ctx_mask << 8;
|
|
ixp2000_uengine_csr_write(uengine, CTX_ENABLES, mask);
|
|
}
|
|
EXPORT_SYMBOL(ixp2000_uengine_start_contexts);
|
|
|
|
void ixp2000_uengine_stop_contexts(int uengine, u8 ctx_mask)
|
|
{
|
|
u32 mask;
|
|
|
|
/*
|
|
* Disable the Ready->Executing transition. Note that this
|
|
* does not stop the context until it voluntarily yields.
|
|
*/
|
|
mask = ixp2000_uengine_csr_read(uengine, CTX_ENABLES);
|
|
mask &= ~(ctx_mask << 8);
|
|
ixp2000_uengine_csr_write(uengine, CTX_ENABLES, mask);
|
|
}
|
|
EXPORT_SYMBOL(ixp2000_uengine_stop_contexts);
|
|
|
|
static int check_ixp_type(struct ixp2000_uengine_code *c)
|
|
{
|
|
u32 product_id;
|
|
u32 rev;
|
|
|
|
product_id = ixp2000_reg_read(IXP2000_PRODUCT_ID);
|
|
if (((product_id >> 16) & 0x1f) != 0)
|
|
return 0;
|
|
|
|
switch ((product_id >> 8) & 0xff) {
|
|
case 0: /* IXP2800 */
|
|
if (!(c->cpu_model_bitmask & 4))
|
|
return 0;
|
|
break;
|
|
|
|
case 1: /* IXP2850 */
|
|
if (!(c->cpu_model_bitmask & 8))
|
|
return 0;
|
|
break;
|
|
|
|
case 2: /* IXP2400 */
|
|
if (!(c->cpu_model_bitmask & 2))
|
|
return 0;
|
|
break;
|
|
|
|
default:
|
|
return 0;
|
|
}
|
|
|
|
rev = product_id & 0xff;
|
|
if (rev < c->cpu_min_revision || rev > c->cpu_max_revision)
|
|
return 0;
|
|
|
|
return 1;
|
|
}
|
|
|
|
static void generate_ucode(u8 *ucode, u32 *gpr_a, u32 *gpr_b)
|
|
{
|
|
int offset;
|
|
int i;
|
|
|
|
offset = 0;
|
|
|
|
for (i = 0; i < 128; i++) {
|
|
u8 b3;
|
|
u8 b2;
|
|
u8 b1;
|
|
u8 b0;
|
|
|
|
b3 = (gpr_a[i] >> 24) & 0xff;
|
|
b2 = (gpr_a[i] >> 16) & 0xff;
|
|
b1 = (gpr_a[i] >> 8) & 0xff;
|
|
b0 = gpr_a[i] & 0xff;
|
|
|
|
// immed[@ai, (b1 << 8) | b0]
|
|
// 11110000 0000VVVV VVVV11VV VVVVVV00 1IIIIIII
|
|
ucode[offset++] = 0xf0;
|
|
ucode[offset++] = (b1 >> 4);
|
|
ucode[offset++] = (b1 << 4) | 0x0c | (b0 >> 6);
|
|
ucode[offset++] = (b0 << 2);
|
|
ucode[offset++] = 0x80 | i;
|
|
|
|
// immed_w1[@ai, (b3 << 8) | b2]
|
|
// 11110100 0100VVVV VVVV11VV VVVVVV00 1IIIIIII
|
|
ucode[offset++] = 0xf4;
|
|
ucode[offset++] = 0x40 | (b3 >> 4);
|
|
ucode[offset++] = (b3 << 4) | 0x0c | (b2 >> 6);
|
|
ucode[offset++] = (b2 << 2);
|
|
ucode[offset++] = 0x80 | i;
|
|
}
|
|
|
|
for (i = 0; i < 128; i++) {
|
|
u8 b3;
|
|
u8 b2;
|
|
u8 b1;
|
|
u8 b0;
|
|
|
|
b3 = (gpr_b[i] >> 24) & 0xff;
|
|
b2 = (gpr_b[i] >> 16) & 0xff;
|
|
b1 = (gpr_b[i] >> 8) & 0xff;
|
|
b0 = gpr_b[i] & 0xff;
|
|
|
|
// immed[@bi, (b1 << 8) | b0]
|
|
// 11110000 0000VVVV VVVV001I IIIIII11 VVVVVVVV
|
|
ucode[offset++] = 0xf0;
|
|
ucode[offset++] = (b1 >> 4);
|
|
ucode[offset++] = (b1 << 4) | 0x02 | (i >> 6);
|
|
ucode[offset++] = (i << 2) | 0x03;
|
|
ucode[offset++] = b0;
|
|
|
|
// immed_w1[@bi, (b3 << 8) | b2]
|
|
// 11110100 0100VVVV VVVV001I IIIIII11 VVVVVVVV
|
|
ucode[offset++] = 0xf4;
|
|
ucode[offset++] = 0x40 | (b3 >> 4);
|
|
ucode[offset++] = (b3 << 4) | 0x02 | (i >> 6);
|
|
ucode[offset++] = (i << 2) | 0x03;
|
|
ucode[offset++] = b2;
|
|
}
|
|
|
|
// ctx_arb[kill]
|
|
ucode[offset++] = 0xe0;
|
|
ucode[offset++] = 0x00;
|
|
ucode[offset++] = 0x01;
|
|
ucode[offset++] = 0x00;
|
|
ucode[offset++] = 0x00;
|
|
}
|
|
|
|
static int set_initial_registers(int uengine, struct ixp2000_uengine_code *c)
|
|
{
|
|
int per_ctx_regs;
|
|
u32 *gpr_a;
|
|
u32 *gpr_b;
|
|
u8 *ucode;
|
|
int i;
|
|
|
|
gpr_a = kmalloc(128 * sizeof(u32), GFP_KERNEL);
|
|
gpr_b = kmalloc(128 * sizeof(u32), GFP_KERNEL);
|
|
ucode = kmalloc(513 * 5, GFP_KERNEL);
|
|
if (gpr_a == NULL || gpr_b == NULL || ucode == NULL) {
|
|
kfree(ucode);
|
|
kfree(gpr_b);
|
|
kfree(gpr_a);
|
|
return 1;
|
|
}
|
|
|
|
per_ctx_regs = 16;
|
|
if (c->uengine_parameters & IXP2000_UENGINE_4_CONTEXTS)
|
|
per_ctx_regs = 32;
|
|
|
|
memset(gpr_a, 0, sizeof(gpr_a));
|
|
memset(gpr_b, 0, sizeof(gpr_b));
|
|
for (i = 0; i < 256; i++) {
|
|
struct ixp2000_reg_value *r = c->initial_reg_values + i;
|
|
u32 *bank;
|
|
int inc;
|
|
int j;
|
|
|
|
if (r->reg == -1)
|
|
break;
|
|
|
|
bank = (r->reg & 0x400) ? gpr_b : gpr_a;
|
|
inc = (r->reg & 0x80) ? 128 : per_ctx_regs;
|
|
|
|
j = r->reg & 0x7f;
|
|
while (j < 128) {
|
|
bank[j] = r->value;
|
|
j += inc;
|
|
}
|
|
}
|
|
|
|
generate_ucode(ucode, gpr_a, gpr_b);
|
|
ixp2000_uengine_load_microcode(uengine, ucode, 513);
|
|
ixp2000_uengine_init_context(uengine, 0, 0);
|
|
ixp2000_uengine_start_contexts(uengine, 0x01);
|
|
for (i = 0; i < 100; i++) {
|
|
u32 status;
|
|
|
|
status = ixp2000_uengine_csr_read(uengine, ACTIVE_CTX_STS);
|
|
if (!(status & 0x80000000))
|
|
break;
|
|
}
|
|
ixp2000_uengine_stop_contexts(uengine, 0x01);
|
|
|
|
kfree(ucode);
|
|
kfree(gpr_b);
|
|
kfree(gpr_a);
|
|
|
|
return !!(i == 100);
|
|
}
|
|
|
|
int ixp2000_uengine_load(int uengine, struct ixp2000_uengine_code *c)
|
|
{
|
|
int ctx;
|
|
|
|
if (!check_ixp_type(c))
|
|
return 1;
|
|
|
|
if (!(ixp2000_uengine_mask & (1 << uengine)))
|
|
return 1;
|
|
|
|
ixp2000_uengine_reset(1 << uengine);
|
|
ixp2000_uengine_set_mode(uengine, c->uengine_parameters);
|
|
if (set_initial_registers(uengine, c))
|
|
return 1;
|
|
ixp2000_uengine_load_microcode(uengine, c->insns, c->num_insns);
|
|
|
|
for (ctx = 0; ctx < 8; ctx++)
|
|
ixp2000_uengine_init_context(uengine, ctx, 0);
|
|
|
|
return 0;
|
|
}
|
|
EXPORT_SYMBOL(ixp2000_uengine_load);
|
|
|
|
|
|
static int __init ixp2000_uengine_init(void)
|
|
{
|
|
int uengine;
|
|
u32 value;
|
|
|
|
/*
|
|
* Determine number of microengines present.
|
|
*/
|
|
switch ((ixp2000_reg_read(IXP2000_PRODUCT_ID) >> 8) & 0x1fff) {
|
|
case 0: /* IXP2800 */
|
|
case 1: /* IXP2850 */
|
|
ixp2000_uengine_mask = 0x00ff00ff;
|
|
break;
|
|
|
|
case 2: /* IXP2400 */
|
|
ixp2000_uengine_mask = 0x000f000f;
|
|
break;
|
|
|
|
default:
|
|
printk(KERN_INFO "Detected unknown IXP2000 model (%.8x)\n",
|
|
(unsigned int)ixp2000_reg_read(IXP2000_PRODUCT_ID));
|
|
ixp2000_uengine_mask = 0x00000000;
|
|
break;
|
|
}
|
|
|
|
/*
|
|
* Reset microengines.
|
|
*/
|
|
ixp2000_uengine_reset(ixp2000_uengine_mask);
|
|
|
|
/*
|
|
* Synchronise timestamp counters across all microengines.
|
|
*/
|
|
value = ixp2000_reg_read(IXP2000_MISC_CONTROL);
|
|
ixp2000_reg_wrb(IXP2000_MISC_CONTROL, value & ~0x80);
|
|
for (uengine = 0; uengine < 32; uengine++) {
|
|
if (ixp2000_uengine_mask & (1 << uengine)) {
|
|
ixp2000_uengine_csr_write(uengine, TIMESTAMP_LOW, 0);
|
|
ixp2000_uengine_csr_write(uengine, TIMESTAMP_HIGH, 0);
|
|
}
|
|
}
|
|
ixp2000_reg_wrb(IXP2000_MISC_CONTROL, value | 0x80);
|
|
|
|
return 0;
|
|
}
|
|
|
|
subsys_initcall(ixp2000_uengine_init);
|