/***********************license start*********************************** * Copyright (c) 2003-2017 Cavium Inc. (support@cavium.com). All rights * reserved. * * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are * met: * * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * * 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. * * * Neither the name of Cavium Inc. nor the names of * its contributors may be used to endorse or promote products * derived from this software without specific prior written * permission. * * This Software, including technical data, may be subject to U.S. export * control laws, including the U.S. Export Administration Act and its * associated regulations, and may be subject to export or import * regulations in other countries. * * TO THE MAXIMUM EXTENT PERMITTED BY LAW, THE SOFTWARE IS PROVIDED "AS IS" * AND WITH ALL FAULTS AND CAVIUM INC. MAKES NO PROMISES, REPRESENTATIONS OR * WARRANTIES, EITHER EXPRESS, IMPLIED, STATUTORY, OR OTHERWISE, WITH RESPECT * TO THE SOFTWARE, INCLUDING ITS CONDITION, ITS CONFORMITY TO ANY * REPRESENTATION OR DESCRIPTION, OR THE EXISTENCE OF ANY LATENT OR PATENT * DEFECTS, AND CAVIUM SPECIFICALLY DISCLAIMS ALL IMPLIED (IF ANY) WARRANTIES * OF TITLE, MERCHANTABILITY, NONINFRINGEMENT, FITNESS FOR A PARTICULAR * PURPOSE, LACK OF VIRUSES, ACCURACY OR COMPLETENESS, QUIET ENJOYMENT, * QUIET POSSESSION OR CORRESPONDENCE TO DESCRIPTION. THE ENTIRE RISK * ARISING OUT OF USE OR PERFORMANCE OF THE SOFTWARE LIES WITH YOU. ***********************license end**************************************/ #include #include #include #include /* This code is an optional part of the BDK. It is only linked in * if BDK_REQUIRE() needs it */ //BDK_REQUIRE(TWSI); BDK_REQUIRE_DEFINE(XFI); /* Rate Select Settings Mode State : 6/8 Rate Select State : 0 RSEL1 : 0 RSEL0 : 0 Ref Clock Gen(MHz) : 156.25 Data Rate(Gbps) : 10.3125 Description : 10 GbE Data Rate Detection Configuration Registers Mode Pin Settings: Mode State : 0 MODE1 : 0 MODE0 : 0 Mode : Two-wire serial interface mode LOS Pin Strap Mode Settings Mode State : 2/6/8 State : 4 LOS1 : Float LOS0 : Float LOS Amplitude(mVpp) : 20 LOS Hysteresis(dB) : 2 Input Equalization Retimer Mode Settings Mode State : 6/8 EQ State : 0 EQ1 : 0 EQ0 : 0 EQ(dB) : Auto DFE : Auto Comment : Full Auto Input Equalization Re-Driver Mode Settings Mode State : EQ State : 0 EQ1 : 0 EQ0 : 0 EQ(dB) : Auto DFE : APowered Down Comment : Analog EQ Only Output De-Emphasis Retimer Mode Settings Mode State : 6/8 DE State : 3 TX1 : Float TX0 : 0 PRE c(-1) mA : -1 MAIN c( 0) mA : 15 POST c(+1) mA : 4 DC Amplitude(mV): 500 De-Emphasis(dB) : -6.02 Comment : Output De-Emphasis Re-Driver Mode Settings Mode State : 2 DE State : 3 TX1 : Float TX0 : 0 Frequency(Gbps) : 10.3125 DC Amplitude(mV): 600 De-Emphasis(dB) : 4 Comment : 10GbE */ static int debug = 0; #define xfi_printf(fmt, args...) \ do { \ if(debug == 1){ \ printf(fmt, ##args); \ } \ } while(0) int bdk_xfi_vsc7224_dump(int twsi_id, int unit){ bdk_node_t node=0; uint8_t dev_addr=0x10 + unit; uint16_t internal_addr=0x7F; int num_bytes=2; int ia_width_bytes=1; uint64_t data=0; int p, i; uint64_t result[0x100] = {0}; uint64_t pagenum[9] = {0x00, 0x01, 0x02, 0x03, 0x20, 0x21, 0x30, 0x31, 0x40}; for(p=0; p < (sizeof(pagenum)/sizeof(pagenum[0])); p++){ data = pagenum[p]; bdk_twsix_write_ia(node, twsi_id, dev_addr, internal_addr, num_bytes, ia_width_bytes, data); for(i=0x80; i<=0xFF; i++){ result[i] = 0x00; result[i] = bdk_twsix_read_ia(node, twsi_id, dev_addr, (uint16_t)i, num_bytes, ia_width_bytes); } for(i=0x80; i<=0xFF; i++){ if(i==0x80){ printf("\npage_%02X[0x100] = {\n", (uint8_t)pagenum[p]); } if(i % 8 == 0){ printf("/* 0x%2X */", i); } printf(" 0x%04X,", (uint16_t)result[i]); if(i==0xFF){ printf("};"); } if((i+1) % 8 == 0){ printf("\n"); } } } return 0; } /* XFI ReTimer/ReDriver Mode Settings */ /* power down regs: Page Reg Position Mask val RegFieldName 0x00 0x89 b07 0x0080 1 PD_INBUF 0x00 0x8A b10 0x0400 1 PD_DFECRU 0x00 0x8A b01 0x0002 1 PD_DFE 0x00 0x8A b00 0x0001 1 PD_DFEADAPT 0x00 0x97 b15 0x8000 1 ASYN_SYNN 0x00 0x97 b09 0x0200 1 PD_OD 0x00 0xA0 b11 0x0800 1 PD_LOS 0x00 0xA4 b15 0x8000 1 PD_CH 0x00 0xB5 b07 0x0080 1 PD_INBUF 0x00 0xB9 b15 0x8000 1 ASYN_SYNN 0x00 0xB9 b09 0x0200 1 PD_OD 0x00 0xBF b07 0x0080 1 PD_INBUF 0x00 0xF0 b15 0x8000 1 ASYN_SYNN 0x00 0xF0 b09 0x0200 1 PD_OD 0x00 0xF6 b07 0x0080 1 PD_INBUF 0x00 0xFA b15 0x8000 1 ASYN_SYNN 0x00 0xFA b09 0x0200 1 PD_OD */ struct regmap{ short int page; unsigned char reg; unsigned short int retimer; unsigned short int redriver; }; /* This table only applies to SFF8104 */ struct regmap xfiregmap[64] = { //CH 0 {0x00, 0x84, 0x0800, 0x0000}, //EQTABLE_DCOFF0 (0n_84) {0x00, 0x8A, 0x7000, 0x0400}, //DFECRU_CTRL (0n_8A) {0x00, 0x8B, 0x4060, 0x0000}, //DFECRU_CFVF_CFAP (0n_8B) {0x00, 0x90, 0xDE85, 0x0000}, //DFECRU_DFEAUTO (0n_90) {0x00, 0x91, 0x2020, 0x0000}, //DFECRU_BTMX_BFMX (0n_91) {0x00, 0x92, 0x0860, 0x0000}, //DFECRU_DXMX_TRMX (0n_92) {0x00, 0x93, 0x6000, 0x0000}, //DFECRU_TRMN_ERRI (0n_93) {0x00, 0x94, 0x0001, 0x0000}, //DFECRU_DFEMODE (0n_94) {0x00, 0x95, 0x0008, 0x0000}, //DFECRU_RATESEL (0n_95) {0x00, 0x97, 0x0000, 0x8080}, //OUTDRVCTRL (0n_97) {0x00, 0x99, 0x001E, 0x0014}, //KR_MAINTAP (0n_99) {0x00, 0x9A, 0x000B, 0x0000}, //KR_PRETAP (0n_9A) {0x00, 0x9B, 0x0010, 0x0000}, //KR_POSTTAP (0n_9B) {0x00, 0x9E, 0x03E8, 0x07D0}, //LOSASSRT (0n_9E) {0x00, 0x9F, 0x04EA, 0x09D5}, //LOSDASSRT (0n_9F) {0x00, 0xB2, 0x0888, 0x0000}, //NA //CH 1 {0x01, 0x84, 0x0800, 0x0000}, {0x01, 0x8A, 0x7000, 0x0400}, {0x01, 0x8B, 0x4060, 0x0000}, {0x01, 0x90, 0xDE85, 0x0000}, {0x01, 0x91, 0x2020, 0x0000}, {0x01, 0x92, 0x0860, 0x0000}, {0x01, 0x93, 0x6000, 0x0000}, {0x01, 0x94, 0x0001, 0x0000}, {0x01, 0x95, 0x0008, 0x0000}, {0x01, 0x97, 0x0000, 0x8080}, {0x01, 0x99, 0x001E, 0x0014}, {0x01, 0x9A, 0x000B, 0x0000}, {0x01, 0x9B, 0x0010, 0x0000}, {0x01, 0x9E, 0x03E8, 0x07D0}, {0x01, 0x9F, 0x04EA, 0x09D5}, {0x01, 0xB2, 0x0888, 0x0000}, //POWER_DOWN Channel 2 and 3 {0x02, 0x8A, 0x0400, 0x0400}, {0x02, 0xA4, 0x8000, 0x8000}, {0x03, 0x8A, 0x0400, 0x0400}, {0x03, 0xA4, 0x8000, 0x8000}, {0x30, 0x80, 0x3453, 0x0000}, //FSYNM_NVAL (3f_80) {0x30, 0x81, 0x00F6, 0x0000}, //FSYNFVAL_MSB (3f_81) {0x30, 0x82, 0x8800, 0x0000}, //FSYNFVAL_LSB (3f_82) {0x30, 0x83, 0x000F, 0x0000}, //FSYNRVAL_MSB (3f_83) {0x30, 0x84, 0xB5E0, 0x0000}, //FSYNRVAL_LSB (3f_84) {0x30, 0x85, 0x0000, 0x0400}, //FSYNTST (3f_85) {0x40, 0x80, 0x4C00, 0x0000}, //ANMUXSEL (40_80) {0x40, 0x81, 0x4000, 0x0000}, //DGMUXCTRL (40_81) {0x40, 0x82, 0x7800, 0xC000}, //RCKINCTRL (40_82) {0x40, 0x84, 0x0020, 0x0000}, //CHRCKSEL (40_84) {-1, 0, 0, 0}, }; int bdk_vsc7224_modeset(int twsi_id, int unit, int xfi_mode){ bdk_node_t node=0; uint8_t dev_addr=0x10 + unit; uint16_t internal_addr=0x7F; uint16_t page=0; int num_bytes=2; int ia_width_bytes=1; uint64_t data=0; int val=0; int ret = 0, r=0; uint16_t reg = 0; if(xfi_mode==0){ printf("XFI Mode Retimer\n"); }else{ printf("XFI Mode Redriver\n"); } while(xfiregmap[r].page != -1){ page = xfiregmap[r].page; reg = xfiregmap[r].reg; if(xfi_mode==0){ data = xfiregmap[r].retimer; }else{ data = xfiregmap[r].redriver; } ret = bdk_twsix_write_ia(node, twsi_id, dev_addr, internal_addr, num_bytes, ia_width_bytes, (uint64_t)page); if(ret !=0){ printf("XFI init Error\n"); break; } ret = bdk_twsix_write_ia(node, twsi_id, dev_addr, reg, num_bytes, ia_width_bytes, data); if(ret !=0){ printf("XFI init Error\n"); break; } val = bdk_twsix_read_ia(node, twsi_id, dev_addr, reg, num_bytes, ia_width_bytes); if(val == -1){ printf("XFI Read Reg Failed @ page:reg :: %2X:%2X \n",page, reg); break; }else{ xfi_printf(" Page: reg: data: val :: %2X:%2X:%04X:%04X\n", page, reg, (uint16_t)data, val); } r++; } return ret; } int bdk_vsc7224_regmap_modeget(int twsi_id, int unit){ bdk_node_t node=0; uint8_t dev_addr=0x10 + unit; uint16_t internal_addr=0x7F; uint16_t page=0; int num_bytes=2; int ia_width_bytes=1; //uint64_t data=0; uint16_t reg = 0; int ret = 0, r=0; int data; printf("\n===========================================\n"); printf("Page :Reg :Value :Retimer :Redriver\n"); printf("===========================================\n"); while(xfiregmap[r].page != -1){ page = xfiregmap[r].page; reg = xfiregmap[r].reg; ret = bdk_twsix_write_ia(node, twsi_id, dev_addr, internal_addr, num_bytes, ia_width_bytes, (uint64_t)page); if(ret !=0){ printf("XFI init Error\n"); break; } data = bdk_twsix_read_ia(node, twsi_id, dev_addr, reg, num_bytes, ia_width_bytes); if(data == -1){ printf("XFI Read Reg Failed @ page:reg :: %2X:%2X \n",page, reg); break; } printf(" %02X: %02X: %04X: %04X: %04X\n", page, reg, (uint16_t)data, xfiregmap[r].retimer, xfiregmap[r].redriver); r++; } printf("=======================================\n"); return ret; } int bdk_vsc7224_wp_regs(int twsi_id, int unit, int xfi_wp){ bdk_node_t node=0; uint8_t dev_addr=0x10 + unit; uint16_t internal_addr=0x7E; uint16_t data=0x0000; int num_bytes=2; int ia_width_bytes=1; int ret =0; if(xfi_wp == 1){ data = 0x0000; }else{ data = 0xFFFF; } ret = bdk_twsix_write_ia(node, twsi_id, dev_addr, internal_addr, num_bytes, ia_width_bytes, (uint64_t)data); if(ret !=0){ printf("XFI VSC7224 Write Protect Error\n"); } return ret; } int bdk_vsc7224_set_reg(int twsi_id, int unit, int page, int reg, int val){ bdk_node_t node=0; uint8_t dev_addr=0x10 + unit; uint16_t internal_addr = reg; int num_bytes=2; int ia_width_bytes=1; int ret=0; xfi_printf(" Unit: Page: reg: val :: %02x:%2X:%2X:%04X\n", unit, page, reg, val & 0xFFFF); ret = bdk_twsix_write_ia(node, twsi_id, dev_addr, 0x7F, num_bytes, ia_width_bytes, (uint64_t)(page & 0xFF)); if (ret) { printf("XFI VSC7224 TWSI Set Page Register Error\n"); } ret = bdk_twsix_write_ia(node, twsi_id, dev_addr, internal_addr, num_bytes, ia_width_bytes, (uint64_t)(val & 0xFFFF)); if (ret) { printf("XFI VSC7224 TWSI Set Register Error\n"); } return ret; } int bdk_vsc7224_debug(int _debug){ debug =_debug; return 0; }