368 lines
12 KiB
C
368 lines
12 KiB
C
/*
|
|
* Copyright (c) 2003-2012 Broadcom Corporation
|
|
* All Rights Reserved
|
|
*
|
|
* This software is available to you under a choice of one of two
|
|
* licenses. You may choose to be licensed under the terms of the GNU
|
|
* General Public License (GPL) Version 2, available from the file
|
|
* COPYING in the main directory of this source tree, or the Broadcom
|
|
* license below:
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
*
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in
|
|
* the documentation and/or other materials provided with the
|
|
* distribution.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY BROADCOM ``AS IS'' AND ANY EXPRESS OR
|
|
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL BROADCOM OR CONTRIBUTORS BE LIABLE
|
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
|
|
* BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
|
|
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE
|
|
* OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN
|
|
* IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
#ifndef _NLM_FMN_H_
|
|
#define _NLM_FMN_H_
|
|
|
|
#include <asm/netlogic/mips-extns.h> /* for COP2 access */
|
|
|
|
/* Station IDs */
|
|
#define FMN_STNID_CPU0 0x00
|
|
#define FMN_STNID_CPU1 0x08
|
|
#define FMN_STNID_CPU2 0x10
|
|
#define FMN_STNID_CPU3 0x18
|
|
#define FMN_STNID_CPU4 0x20
|
|
#define FMN_STNID_CPU5 0x28
|
|
#define FMN_STNID_CPU6 0x30
|
|
#define FMN_STNID_CPU7 0x38
|
|
|
|
#define FMN_STNID_XGS0_TX 64
|
|
#define FMN_STNID_XMAC0_00_TX 64
|
|
#define FMN_STNID_XMAC0_01_TX 65
|
|
#define FMN_STNID_XMAC0_02_TX 66
|
|
#define FMN_STNID_XMAC0_03_TX 67
|
|
#define FMN_STNID_XMAC0_04_TX 68
|
|
#define FMN_STNID_XMAC0_05_TX 69
|
|
#define FMN_STNID_XMAC0_06_TX 70
|
|
#define FMN_STNID_XMAC0_07_TX 71
|
|
#define FMN_STNID_XMAC0_08_TX 72
|
|
#define FMN_STNID_XMAC0_09_TX 73
|
|
#define FMN_STNID_XMAC0_10_TX 74
|
|
#define FMN_STNID_XMAC0_11_TX 75
|
|
#define FMN_STNID_XMAC0_12_TX 76
|
|
#define FMN_STNID_XMAC0_13_TX 77
|
|
#define FMN_STNID_XMAC0_14_TX 78
|
|
#define FMN_STNID_XMAC0_15_TX 79
|
|
|
|
#define FMN_STNID_XGS1_TX 80
|
|
#define FMN_STNID_XMAC1_00_TX 80
|
|
#define FMN_STNID_XMAC1_01_TX 81
|
|
#define FMN_STNID_XMAC1_02_TX 82
|
|
#define FMN_STNID_XMAC1_03_TX 83
|
|
#define FMN_STNID_XMAC1_04_TX 84
|
|
#define FMN_STNID_XMAC1_05_TX 85
|
|
#define FMN_STNID_XMAC1_06_TX 86
|
|
#define FMN_STNID_XMAC1_07_TX 87
|
|
#define FMN_STNID_XMAC1_08_TX 88
|
|
#define FMN_STNID_XMAC1_09_TX 89
|
|
#define FMN_STNID_XMAC1_10_TX 90
|
|
#define FMN_STNID_XMAC1_11_TX 91
|
|
#define FMN_STNID_XMAC1_12_TX 92
|
|
#define FMN_STNID_XMAC1_13_TX 93
|
|
#define FMN_STNID_XMAC1_14_TX 94
|
|
#define FMN_STNID_XMAC1_15_TX 95
|
|
|
|
#define FMN_STNID_GMAC 96
|
|
#define FMN_STNID_GMACJFR_0 96
|
|
#define FMN_STNID_GMACRFR_0 97
|
|
#define FMN_STNID_GMACTX0 98
|
|
#define FMN_STNID_GMACTX1 99
|
|
#define FMN_STNID_GMACTX2 100
|
|
#define FMN_STNID_GMACTX3 101
|
|
#define FMN_STNID_GMACJFR_1 102
|
|
#define FMN_STNID_GMACRFR_1 103
|
|
|
|
#define FMN_STNID_DMA 104
|
|
#define FMN_STNID_DMA_0 104
|
|
#define FMN_STNID_DMA_1 105
|
|
#define FMN_STNID_DMA_2 106
|
|
#define FMN_STNID_DMA_3 107
|
|
|
|
#define FMN_STNID_XGS0FR 112
|
|
#define FMN_STNID_XMAC0JFR 112
|
|
#define FMN_STNID_XMAC0RFR 113
|
|
|
|
#define FMN_STNID_XGS1FR 114
|
|
#define FMN_STNID_XMAC1JFR 114
|
|
#define FMN_STNID_XMAC1RFR 115
|
|
#define FMN_STNID_SEC 120
|
|
#define FMN_STNID_SEC0 120
|
|
#define FMN_STNID_SEC1 121
|
|
#define FMN_STNID_SEC2 122
|
|
#define FMN_STNID_SEC3 123
|
|
#define FMN_STNID_PK0 124
|
|
#define FMN_STNID_SEC_RSA 124
|
|
#define FMN_STNID_SEC_RSVD0 125
|
|
#define FMN_STNID_SEC_RSVD1 126
|
|
#define FMN_STNID_SEC_RSVD2 127
|
|
|
|
#define FMN_STNID_GMAC1 80
|
|
#define FMN_STNID_GMAC1_FR_0 81
|
|
#define FMN_STNID_GMAC1_TX0 82
|
|
#define FMN_STNID_GMAC1_TX1 83
|
|
#define FMN_STNID_GMAC1_TX2 84
|
|
#define FMN_STNID_GMAC1_TX3 85
|
|
#define FMN_STNID_GMAC1_FR_1 87
|
|
#define FMN_STNID_GMAC0 96
|
|
#define FMN_STNID_GMAC0_FR_0 97
|
|
#define FMN_STNID_GMAC0_TX0 98
|
|
#define FMN_STNID_GMAC0_TX1 99
|
|
#define FMN_STNID_GMAC0_TX2 100
|
|
#define FMN_STNID_GMAC0_TX3 101
|
|
#define FMN_STNID_GMAC0_FR_1 103
|
|
#define FMN_STNID_CMP_0 108
|
|
#define FMN_STNID_CMP_1 109
|
|
#define FMN_STNID_CMP_2 110
|
|
#define FMN_STNID_CMP_3 111
|
|
#define FMN_STNID_PCIE_0 116
|
|
#define FMN_STNID_PCIE_1 117
|
|
#define FMN_STNID_PCIE_2 118
|
|
#define FMN_STNID_PCIE_3 119
|
|
#define FMN_STNID_XLS_PK0 121
|
|
|
|
#define nlm_read_c2_cc0(s) __read_32bit_c2_register($16, s)
|
|
#define nlm_read_c2_cc1(s) __read_32bit_c2_register($17, s)
|
|
#define nlm_read_c2_cc2(s) __read_32bit_c2_register($18, s)
|
|
#define nlm_read_c2_cc3(s) __read_32bit_c2_register($19, s)
|
|
#define nlm_read_c2_cc4(s) __read_32bit_c2_register($20, s)
|
|
#define nlm_read_c2_cc5(s) __read_32bit_c2_register($21, s)
|
|
#define nlm_read_c2_cc6(s) __read_32bit_c2_register($22, s)
|
|
#define nlm_read_c2_cc7(s) __read_32bit_c2_register($23, s)
|
|
#define nlm_read_c2_cc8(s) __read_32bit_c2_register($24, s)
|
|
#define nlm_read_c2_cc9(s) __read_32bit_c2_register($25, s)
|
|
#define nlm_read_c2_cc10(s) __read_32bit_c2_register($26, s)
|
|
#define nlm_read_c2_cc11(s) __read_32bit_c2_register($27, s)
|
|
#define nlm_read_c2_cc12(s) __read_32bit_c2_register($28, s)
|
|
#define nlm_read_c2_cc13(s) __read_32bit_c2_register($29, s)
|
|
#define nlm_read_c2_cc14(s) __read_32bit_c2_register($30, s)
|
|
#define nlm_read_c2_cc15(s) __read_32bit_c2_register($31, s)
|
|
|
|
#define nlm_write_c2_cc0(s, v) __write_32bit_c2_register($16, s, v)
|
|
#define nlm_write_c2_cc1(s, v) __write_32bit_c2_register($17, s, v)
|
|
#define nlm_write_c2_cc2(s, v) __write_32bit_c2_register($18, s, v)
|
|
#define nlm_write_c2_cc3(s, v) __write_32bit_c2_register($19, s, v)
|
|
#define nlm_write_c2_cc4(s, v) __write_32bit_c2_register($20, s, v)
|
|
#define nlm_write_c2_cc5(s, v) __write_32bit_c2_register($21, s, v)
|
|
#define nlm_write_c2_cc6(s, v) __write_32bit_c2_register($22, s, v)
|
|
#define nlm_write_c2_cc7(s, v) __write_32bit_c2_register($23, s, v)
|
|
#define nlm_write_c2_cc8(s, v) __write_32bit_c2_register($24, s, v)
|
|
#define nlm_write_c2_cc9(s, v) __write_32bit_c2_register($25, s, v)
|
|
#define nlm_write_c2_cc10(s, v) __write_32bit_c2_register($26, s, v)
|
|
#define nlm_write_c2_cc11(s, v) __write_32bit_c2_register($27, s, v)
|
|
#define nlm_write_c2_cc12(s, v) __write_32bit_c2_register($28, s, v)
|
|
#define nlm_write_c2_cc13(s, v) __write_32bit_c2_register($29, s, v)
|
|
#define nlm_write_c2_cc14(s, v) __write_32bit_c2_register($30, s, v)
|
|
#define nlm_write_c2_cc15(s, v) __write_32bit_c2_register($31, s, v)
|
|
|
|
#define nlm_read_c2_status0() __read_32bit_c2_register($2, 0)
|
|
#define nlm_write_c2_status0(v) __write_32bit_c2_register($2, 0, v)
|
|
#define nlm_read_c2_status1() __read_32bit_c2_register($2, 1)
|
|
#define nlm_write_c2_status1(v) __write_32bit_c2_register($2, 1, v)
|
|
#define nlm_read_c2_status(sel) __read_32bit_c2_register($2, 0)
|
|
#define nlm_read_c2_config() __read_32bit_c2_register($3, 0)
|
|
#define nlm_write_c2_config(v) __write_32bit_c2_register($3, 0, v)
|
|
#define nlm_read_c2_bucksize(b) __read_32bit_c2_register($4, b)
|
|
#define nlm_write_c2_bucksize(b, v) __write_32bit_c2_register($4, b, v)
|
|
|
|
#define nlm_read_c2_rx_msg0() __read_64bit_c2_register($1, 0)
|
|
#define nlm_read_c2_rx_msg1() __read_64bit_c2_register($1, 1)
|
|
#define nlm_read_c2_rx_msg2() __read_64bit_c2_register($1, 2)
|
|
#define nlm_read_c2_rx_msg3() __read_64bit_c2_register($1, 3)
|
|
|
|
#define nlm_write_c2_tx_msg0(v) __write_64bit_c2_register($0, 0, v)
|
|
#define nlm_write_c2_tx_msg1(v) __write_64bit_c2_register($0, 1, v)
|
|
#define nlm_write_c2_tx_msg2(v) __write_64bit_c2_register($0, 2, v)
|
|
#define nlm_write_c2_tx_msg3(v) __write_64bit_c2_register($0, 3, v)
|
|
|
|
#define FMN_STN_RX_QSIZE 256
|
|
#define FMN_NSTATIONS 128
|
|
#define FMN_CORE_NBUCKETS 8
|
|
|
|
static inline void nlm_msgsnd(unsigned int stid)
|
|
{
|
|
__asm__ volatile (
|
|
".set push\n"
|
|
".set noreorder\n"
|
|
".set noat\n"
|
|
"move $1, %0\n"
|
|
"c2 0x10001\n" /* msgsnd $1 */
|
|
".set pop\n"
|
|
: : "r" (stid) : "$1"
|
|
);
|
|
}
|
|
|
|
static inline void nlm_msgld(unsigned int pri)
|
|
{
|
|
__asm__ volatile (
|
|
".set push\n"
|
|
".set noreorder\n"
|
|
".set noat\n"
|
|
"move $1, %0\n"
|
|
"c2 0x10002\n" /* msgld $1 */
|
|
".set pop\n"
|
|
: : "r" (pri) : "$1"
|
|
);
|
|
}
|
|
|
|
static inline void nlm_msgwait(unsigned int mask)
|
|
{
|
|
__asm__ volatile (
|
|
".set push\n"
|
|
".set noreorder\n"
|
|
".set noat\n"
|
|
"move $8, %0\n"
|
|
"c2 0x10003\n" /* msgwait $1 */
|
|
".set pop\n"
|
|
: : "r" (mask) : "$1"
|
|
);
|
|
}
|
|
|
|
/*
|
|
* Disable interrupts and enable COP2 access
|
|
*/
|
|
static inline uint32_t nlm_cop2_enable_irqsave(void)
|
|
{
|
|
uint32_t sr = read_c0_status();
|
|
|
|
write_c0_status((sr & ~ST0_IE) | ST0_CU2);
|
|
return sr;
|
|
}
|
|
|
|
static inline void nlm_cop2_disable_irqrestore(uint32_t sr)
|
|
{
|
|
write_c0_status(sr);
|
|
}
|
|
|
|
static inline void nlm_fmn_setup_intr(int irq, unsigned int tmask)
|
|
{
|
|
uint32_t config;
|
|
|
|
config = (1 << 24) /* interrupt water mark - 1 msg */
|
|
| (irq << 16) /* irq */
|
|
| (tmask << 8) /* thread mask */
|
|
| 0x2; /* enable watermark intr, disable empty intr */
|
|
nlm_write_c2_config(config);
|
|
}
|
|
|
|
struct nlm_fmn_msg {
|
|
uint64_t msg0;
|
|
uint64_t msg1;
|
|
uint64_t msg2;
|
|
uint64_t msg3;
|
|
};
|
|
|
|
static inline int nlm_fmn_send(unsigned int size, unsigned int code,
|
|
unsigned int stid, struct nlm_fmn_msg *msg)
|
|
{
|
|
unsigned int dest;
|
|
uint32_t status;
|
|
int i;
|
|
|
|
/*
|
|
* Make sure that all the writes pending at the cpu are flushed.
|
|
* Any writes pending on CPU will not be see by devices. L1/L2
|
|
* caches are coherent with IO, so no cache flush needed.
|
|
*/
|
|
__asm __volatile("sync");
|
|
|
|
/* Load TX message buffers */
|
|
nlm_write_c2_tx_msg0(msg->msg0);
|
|
nlm_write_c2_tx_msg1(msg->msg1);
|
|
nlm_write_c2_tx_msg2(msg->msg2);
|
|
nlm_write_c2_tx_msg3(msg->msg3);
|
|
dest = ((size - 1) << 16) | (code << 8) | stid;
|
|
|
|
/*
|
|
* Retry a few times on credit fail, this should be a
|
|
* transient condition, unless there is a configuration
|
|
* failure, or the receiver is stuck.
|
|
*/
|
|
for (i = 0; i < 8; i++) {
|
|
nlm_msgsnd(dest);
|
|
status = nlm_read_c2_status0();
|
|
if ((status & 0x2) == 1)
|
|
pr_info("Send pending fail!\n");
|
|
if ((status & 0x4) == 0)
|
|
return 0;
|
|
}
|
|
|
|
/* If there is a credit failure, return error */
|
|
return status & 0x06;
|
|
}
|
|
|
|
static inline int nlm_fmn_receive(int bucket, int *size, int *code, int *stid,
|
|
struct nlm_fmn_msg *msg)
|
|
{
|
|
uint32_t status, tmp;
|
|
|
|
nlm_msgld(bucket);
|
|
|
|
/* wait for load pending to clear */
|
|
do {
|
|
status = nlm_read_c2_status0();
|
|
} while ((status & 0x08) != 0);
|
|
|
|
/* receive error bits */
|
|
tmp = status & 0x30;
|
|
if (tmp != 0)
|
|
return tmp;
|
|
|
|
*size = ((status & 0xc0) >> 6) + 1;
|
|
*code = (status & 0xff00) >> 8;
|
|
*stid = (status & 0x7f0000) >> 16;
|
|
msg->msg0 = nlm_read_c2_rx_msg0();
|
|
msg->msg1 = nlm_read_c2_rx_msg1();
|
|
msg->msg2 = nlm_read_c2_rx_msg2();
|
|
msg->msg3 = nlm_read_c2_rx_msg3();
|
|
|
|
return 0;
|
|
}
|
|
|
|
struct xlr_fmn_info {
|
|
int num_buckets;
|
|
int start_stn_id;
|
|
int end_stn_id;
|
|
int credit_config[128];
|
|
};
|
|
|
|
struct xlr_board_fmn_config {
|
|
int bucket_size[128]; /* size of buckets for all stations */
|
|
struct xlr_fmn_info cpu[8];
|
|
struct xlr_fmn_info gmac[2];
|
|
struct xlr_fmn_info dma;
|
|
struct xlr_fmn_info cmp;
|
|
struct xlr_fmn_info sae;
|
|
struct xlr_fmn_info xgmac[2];
|
|
};
|
|
|
|
extern int nlm_register_fmn_handler(int start, int end,
|
|
void (*fn)(int, int, int, int, struct nlm_fmn_msg *, void *),
|
|
void *arg);
|
|
extern void xlr_percpu_fmn_init(void);
|
|
extern void nlm_setup_fmn_irq(void);
|
|
extern void xlr_board_info_setup(void);
|
|
|
|
extern struct xlr_board_fmn_config xlr_board_fmn_config;
|
|
#endif
|