mirror of
https://github.com/openwrt/openwrt.git
synced 2025-01-10 15:03:07 +00:00
917 lines
22 KiB
Diff
917 lines
22 KiB
Diff
|
From fa51192b912d296b8eec10f7d44c6c17eb1dd368 Mon Sep 17 00:00:00 2001
|
||
|
From: Xiangfu <xiangfu@openmobilefree.net>
|
||
|
Date: Fri, 12 Oct 2012 09:47:39 +0800
|
||
|
Subject: [PATCH 2/6] qi_lb60: add software usbboot support
|
||
|
|
||
|
JZ4740 CPU have a internal ROM have such kind of code, that make
|
||
|
JZ4740 can boot from USB
|
||
|
|
||
|
usbboot.S can downloads user program from the USB port to internal
|
||
|
SRAM and branches to the internal SRAM to execute the program
|
||
|
|
||
|
Signed-off-by: Xiangfu <xiangfu@openmobilefree.net>
|
||
|
---
|
||
|
board/qi/qi_lb60/Makefile | 1 +
|
||
|
board/qi/qi_lb60/qi_lb60-spl.c | 20 +
|
||
|
board/qi/qi_lb60/usbboot.S | 838 ++++++++++++++++++++++++++++++++++++++++
|
||
|
3 files changed, 859 insertions(+)
|
||
|
create mode 100644 board/qi/qi_lb60/usbboot.S
|
||
|
|
||
|
diff --git a/board/qi/qi_lb60/Makefile b/board/qi/qi_lb60/Makefile
|
||
|
index e399246..6dd8c6f 100644
|
||
|
--- a/board/qi/qi_lb60/Makefile
|
||
|
+++ b/board/qi/qi_lb60/Makefile
|
||
|
@@ -23,6 +23,7 @@ include $(TOPDIR)/config.mk
|
||
|
LIB = $(obj)lib$(BOARD).o
|
||
|
|
||
|
ifeq ($(CONFIG_SPL_BUILD),y)
|
||
|
+SOBJS := usbboot.o
|
||
|
COBJS := $(BOARD)-spl.o
|
||
|
else
|
||
|
COBJS := $(BOARD).o
|
||
|
diff --git a/board/qi/qi_lb60/qi_lb60-spl.c b/board/qi/qi_lb60/qi_lb60-spl.c
|
||
|
index 3fe3fa3..aea459c 100644
|
||
|
--- a/board/qi/qi_lb60/qi_lb60-spl.c
|
||
|
+++ b/board/qi/qi_lb60/qi_lb60-spl.c
|
||
|
@@ -12,6 +12,24 @@
|
||
|
#include <asm/io.h>
|
||
|
#include <asm/jz4740.h>
|
||
|
|
||
|
+#define KEY_U_OUT (32 * 2 + 16)
|
||
|
+#define KEY_U_IN (32 * 3 + 19)
|
||
|
+
|
||
|
+extern void usb_boot(void);
|
||
|
+
|
||
|
+static void check_usb_boot(void)
|
||
|
+{
|
||
|
+ __gpio_as_input(KEY_U_IN);
|
||
|
+ __gpio_enable_pull(KEY_U_IN);
|
||
|
+ __gpio_as_output(KEY_U_OUT);
|
||
|
+ __gpio_clear_pin(KEY_U_OUT);
|
||
|
+
|
||
|
+ if (!__gpio_get_pin(KEY_U_IN)) {
|
||
|
+ puts("[U] pressed, goto USBBOOT mode\n");
|
||
|
+ usb_boot();
|
||
|
+ }
|
||
|
+}
|
||
|
+
|
||
|
void nand_spl_boot(void)
|
||
|
{
|
||
|
__gpio_as_sdram_16bit_4720();
|
||
|
@@ -23,6 +41,8 @@ void nand_spl_boot(void)
|
||
|
pll_init();
|
||
|
sdram_init();
|
||
|
|
||
|
+ check_usb_boot();
|
||
|
+
|
||
|
nand_init();
|
||
|
|
||
|
puts("\nQi LB60 SPL: Starting U-Boot ...\n");
|
||
|
diff --git a/board/qi/qi_lb60/usbboot.S b/board/qi/qi_lb60/usbboot.S
|
||
|
new file mode 100644
|
||
|
index 0000000..c872266
|
||
|
--- /dev/null
|
||
|
+++ b/board/qi/qi_lb60/usbboot.S
|
||
|
@@ -0,0 +1,838 @@
|
||
|
+/*
|
||
|
+ * for jz4740 usb boot
|
||
|
+ *
|
||
|
+ * Copyright (c) 2009 Author: <jlwei@ingenic.cn>
|
||
|
+ *
|
||
|
+ * See file CREDITS for list of people who contributed to this
|
||
|
+ * project.
|
||
|
+ *
|
||
|
+ * This program is free software; you can redistribute it and/or
|
||
|
+ * modify it under the terms of the GNU General Public License as
|
||
|
+ * published by the Free Software Foundation; either version 2 of
|
||
|
+ * the License, or (at your option) any later version.
|
||
|
+ *
|
||
|
+ * This program is distributed in the hope that it will be useful,
|
||
|
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
|
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
|
+ * GNU General Public License for more details.
|
||
|
+ *
|
||
|
+ * You should have received a copy of the GNU General Public License
|
||
|
+ * along with this program; if not, write to the Free Software
|
||
|
+ * Foundation, Inc., 59 Temple Place, Suite 330, Boston,
|
||
|
+ * MA 02111-1307 USA
|
||
|
+ */
|
||
|
+ .set noreorder
|
||
|
+ .globl usb_boot
|
||
|
+ .text
|
||
|
+
|
||
|
+/*
|
||
|
+ * Both NAND and USB boot load data to D-Cache first, then transfer
|
||
|
+ * data from D-Cache to I-Cache, and jump to execute the code in I-Cache.
|
||
|
+ * So init caches first and then dispatch to a proper boot routine.
|
||
|
+ */
|
||
|
+
|
||
|
+.macro load_addr reg addr
|
||
|
+ li \reg, 0x80000000
|
||
|
+ addiu \reg, \reg, \addr
|
||
|
+ la $2, usbboot_begin
|
||
|
+ subu \reg, \reg, $2
|
||
|
+.endm
|
||
|
+
|
||
|
+usb_boot:
|
||
|
+ /* Initialize PLL: set ICLK to 84MHz and HCLK to 42MHz. */
|
||
|
+ la $9, 0xB0000000 /* CPCCR: Clock Control Register */
|
||
|
+ la $8, 0x42041110 /* I:S:M:P=1:2:2:2 */
|
||
|
+ sw $8, 0($9)
|
||
|
+
|
||
|
+ la $9, 0xB0000010 /* CPPCR: PLL Control Register */
|
||
|
+ la $8, 0x06000120 /* M=12 N=0 D=0 CLK=12*(M+2)/(N+2) */
|
||
|
+ sw $8, 0($9)
|
||
|
+
|
||
|
+ mtc0 $0, $26 /* CP0_ERRCTL, restore WST reset state */
|
||
|
+ nop
|
||
|
+
|
||
|
+ mtc0 $0, $16 /* CP0_CONFIG */
|
||
|
+ nop
|
||
|
+
|
||
|
+ /* Relocate code to beginning of the ram */
|
||
|
+
|
||
|
+ la $2, usbboot_begin
|
||
|
+ la $3, usbboot_end
|
||
|
+ li $4, 0x80000000
|
||
|
+
|
||
|
+1:
|
||
|
+ lw $5, 0($2)
|
||
|
+ sw $5, 0($4)
|
||
|
+ addiu $2, $2, 4
|
||
|
+ bne $2, $3, 1b
|
||
|
+ addiu $4, $4, 4
|
||
|
+
|
||
|
+ li $2, 0x80000000
|
||
|
+ ori $3, $2, 0
|
||
|
+ addiu $3, $3, usbboot_end
|
||
|
+ la $4, usbboot_begin
|
||
|
+ subu $3, $3, $4
|
||
|
+
|
||
|
+
|
||
|
+2:
|
||
|
+ cache 0x0, 0($2) /* Index_Invalidate_I */
|
||
|
+ cache 0x1, 0($2) /* Index_Writeback_Inv_D */
|
||
|
+ addiu $2, $2, 32
|
||
|
+ subu $4, $3, $2
|
||
|
+ bgtz $4, 2b
|
||
|
+ nop
|
||
|
+
|
||
|
+ load_addr $3, usb_boot_return
|
||
|
+
|
||
|
+ jr $3
|
||
|
+
|
||
|
+usbboot_begin:
|
||
|
+
|
||
|
+init_caches:
|
||
|
+ li $2, 3 /* cacheable for kseg0 access */
|
||
|
+ mtc0 $2, $16 /* CP0_CONFIG */
|
||
|
+ nop
|
||
|
+
|
||
|
+ li $2, 0x20000000 /* enable idx-store-data cache insn */
|
||
|
+ mtc0 $2, $26 /* CP0_ERRCTL */
|
||
|
+
|
||
|
+ ori $2, $28, 0 /* start address */
|
||
|
+ ori $3, $2, 0x3fe0 /* end address, total 16KB */
|
||
|
+ mtc0 $0, $28, 0 /* CP0_TAGLO */
|
||
|
+ mtc0 $0, $28, 1 /* CP0_DATALO */
|
||
|
+cache_clear_a_line:
|
||
|
+ cache 0x8, 0($2) /* Index_Store_Tag_I */
|
||
|
+ cache 0x9, 0($2) /* Index_Store_Tag_D */
|
||
|
+ bne $2, $3, cache_clear_a_line
|
||
|
+ addiu $2, $2, 32 /* increment CACHE_LINE_SIZE */
|
||
|
+
|
||
|
+ ori $2, $28, 0 /* start address */
|
||
|
+ ori $3, $2, 0x3fe0 /* end address, total 16KB */
|
||
|
+ la $4, 0x1ffff000 /* physical address and 4KB page mask */
|
||
|
+cache_alloc_a_line:
|
||
|
+ and $5, $2, $4
|
||
|
+ ori $5, $5, 1 /* V bit of the physical tag */
|
||
|
+ mtc0 $5, $28, 0 /* CP0_TAGLO */
|
||
|
+ cache 0x8, 0($2) /* Index_Store_Tag_I */
|
||
|
+ cache 0x9, 0($2) /* Index_Store_Tag_D */
|
||
|
+ bne $2, $3, cache_alloc_a_line
|
||
|
+ addiu $2, $2, 32 /* increment CACHE_LINE_SIZE */
|
||
|
+
|
||
|
+ nop
|
||
|
+ nop
|
||
|
+ nop
|
||
|
+ /*
|
||
|
+ * Transfer data from dcache to icache, then jump to icache.
|
||
|
+ * Input parameters:
|
||
|
+ * $19: data length in bytes
|
||
|
+ * $20: jump target address
|
||
|
+ */
|
||
|
+xfer_d2i:
|
||
|
+
|
||
|
+ ori $8, $20, 0
|
||
|
+ addu $9, $8, $19 /* total 16KB */
|
||
|
+
|
||
|
+1:
|
||
|
+ cache 0x0, 0($8) /* Index_Invalidate_I */
|
||
|
+ cache 0x1, 0($8) /* Index_Writeback_Inv_D */
|
||
|
+ bne $8, $9, 1b
|
||
|
+ addiu $8, $8, 32
|
||
|
+
|
||
|
+ /* flush write-buffer */
|
||
|
+ sync
|
||
|
+
|
||
|
+ /* Invalidate BTB */
|
||
|
+ mfc0 $8, $16, 7 /* CP0_CONFIG */
|
||
|
+ nop
|
||
|
+ ori $8, 2
|
||
|
+ mtc0 $8, $16, 7
|
||
|
+ nop
|
||
|
+
|
||
|
+ /* Overwrite config to disable ram initalisation */
|
||
|
+ li $2, 0xff
|
||
|
+ sb $2, 20($20)
|
||
|
+
|
||
|
+ jalr $20
|
||
|
+ nop
|
||
|
+
|
||
|
+icache_return:
|
||
|
+ /* User code can return to here after executing itself in
|
||
|
+ icache, by jumping to $31. */
|
||
|
+ b usb_boot_return
|
||
|
+ nop
|
||
|
+
|
||
|
+
|
||
|
+usb_boot_return:
|
||
|
+ /* Enable the USB PHY */
|
||
|
+ la $9, 0xB0000024 /* CPM_SCR */
|
||
|
+ lw $8, 0($9)
|
||
|
+ ori $8, 0x40 /* USBPHY_ENABLE */
|
||
|
+ sw $8, 0($9)
|
||
|
+
|
||
|
+ /* Initialize USB registers */
|
||
|
+ la $27, 0xb3040000 /* USB registers base address */
|
||
|
+
|
||
|
+ sb $0, 0x0b($27) /* INTRUSBE: disable common USB interrupts */
|
||
|
+ sh $0, 0x06($27) /* INTRINE: disable EPIN interrutps */
|
||
|
+ sh $0, 0x08($27) /* INTROUTE: disable EPOUT interrutps */
|
||
|
+
|
||
|
+ li $9, 0x61
|
||
|
+ sb $9, 0x01($27) /* POWER: HSENAB | SUSPENDM | SOFTCONN */
|
||
|
+
|
||
|
+ /* Initialize USB states */
|
||
|
+ li $22, 0 /* set EP0 to IDLE state */
|
||
|
+ li $23, 1 /* no data stage */
|
||
|
+
|
||
|
+ /* Main loop of polling the usb commands */
|
||
|
+usb_command_loop:
|
||
|
+ lbu $9, 0x0a($27) /* read INTRUSB */
|
||
|
+ andi $9, 0x04 /* check USB_INTR_RESET */
|
||
|
+ beqz $9, check_intr_ep0in
|
||
|
+ nop
|
||
|
+
|
||
|
+ /* 1. Handle USB reset interrupt */
|
||
|
+handle_reset_intr:
|
||
|
+ lbu $9, 0x01($27) /* read POWER */
|
||
|
+ andi $9, 0x10 /* test HS_MODE */
|
||
|
+ bnez $9, _usb_set_maxpktsize
|
||
|
+ li $9, 512 /* max packet size of HS mode */
|
||
|
+ li $9, 64 /* max packet size of FS mode */
|
||
|
+
|
||
|
+_usb_set_maxpktsize:
|
||
|
+ li $8, 1
|
||
|
+ sb $8, 0x0e($27) /* set INDEX 1 */
|
||
|
+
|
||
|
+ sh $9, 0x10($27) /* INMAXP */
|
||
|
+ sb $0, 0x13($27) /* INCSRH */
|
||
|
+ sh $9, 0x14($27) /* OUTMAXP */
|
||
|
+ sb $0, 0x17($27) /* OUTCSRH */
|
||
|
+
|
||
|
+_usb_flush_fifo:
|
||
|
+ li $8, 0x48 /* INCSR_CDT && INCSR_FF */
|
||
|
+ sb $8, 0x12($27) /* INCSR */
|
||
|
+ li $8, 0x90 /* OUTCSR_CDT && OUTCSR_FF */
|
||
|
+ sb $8, 0x16($27) /* OUTCSR */
|
||
|
+
|
||
|
+ li $22, 0 /* set EP0 to IDLE state */
|
||
|
+ li $23, 1 /* no data stage */
|
||
|
+
|
||
|
+ /* 2. Check and handle EP0 interrupt */
|
||
|
+check_intr_ep0in:
|
||
|
+ lhu $10, 0x02($27) /* read INTRIN */
|
||
|
+ andi $9, $10, 0x1 /* check EP0 interrupt */
|
||
|
+ beqz $9, check_intr_ep1in
|
||
|
+ nop
|
||
|
+
|
||
|
+handle_ep0_intr:
|
||
|
+ sb $0, 0x0e($27) /* set INDEX 0 */
|
||
|
+ lbu $11, 0x12($27) /* read CSR0 */
|
||
|
+
|
||
|
+ andi $9, $11, 0x04 /* check SENTSTALL */
|
||
|
+ beqz $9, _ep0_setupend
|
||
|
+ nop
|
||
|
+
|
||
|
+_ep0_sentstall:
|
||
|
+ andi $9, $11, 0xdb
|
||
|
+ sb $9, 0x12($27) /* clear SENDSTALL and SENTSTALL */
|
||
|
+ li $22, 0 /* set EP0 to IDLE state */
|
||
|
+
|
||
|
+_ep0_setupend:
|
||
|
+ andi $9, $11, 0x10 /* check SETUPEND */
|
||
|
+ beqz $9, ep0_idle_state
|
||
|
+ nop
|
||
|
+
|
||
|
+ ori $9, $11, 0x80
|
||
|
+ sb $9, 0x12($27) /* set SVDSETUPEND */
|
||
|
+ li $22, 0 /* set EP0 to IDLE state */
|
||
|
+
|
||
|
+ep0_idle_state:
|
||
|
+ bnez $22, ep0_tx_state
|
||
|
+ nop
|
||
|
+
|
||
|
+ /* 2.1 Handle EP0 IDLE state interrupt */
|
||
|
+ andi $9, $11, 0x01 /* check OUTPKTRDY */
|
||
|
+ beqz $9, check_intr_ep1in
|
||
|
+ nop
|
||
|
+
|
||
|
+ /* Read 8-bytes setup packet from the FIFO */
|
||
|
+ lw $25, 0x20($27) /* first word of setup packet */
|
||
|
+ lw $26, 0x20($27) /* second word of setup packet */
|
||
|
+
|
||
|
+ andi $9, $25, 0x60 /* bRequestType & USB_TYPE_MASK */
|
||
|
+ beqz $9, _ep0_std_req
|
||
|
+ nop
|
||
|
+
|
||
|
+ /* 2.1.1 Vendor-specific setup request */
|
||
|
+_ep0_vend_req:
|
||
|
+ li $22, 0 /* set EP0 to IDLE state */
|
||
|
+ li $23, 1 /* NoData = 1 */
|
||
|
+
|
||
|
+ andi $9, $25, 0xff00 /* check bRequest */
|
||
|
+ srl $9, $9, 8
|
||
|
+ beqz $9, __ep0_get_cpu_info
|
||
|
+ sub $8, $9, 0x1
|
||
|
+ beqz $8, __ep0_set_data_address
|
||
|
+ sub $8, $9, 0x2
|
||
|
+ beqz $8, __ep0_set_data_length
|
||
|
+ sub $8, $9, 0x3
|
||
|
+ beqz $8, __ep0_flush_caches
|
||
|
+ sub $8, $9, 0x4
|
||
|
+ beqz $8, __ep0_prog_start1
|
||
|
+ sub $8, $9, 0x5
|
||
|
+ beqz $8, __ep0_prog_start2
|
||
|
+ nop
|
||
|
+ b _ep0_idle_state_fini /* invalid request */
|
||
|
+ nop
|
||
|
+
|
||
|
+__ep0_get_cpu_info:
|
||
|
+ load_addr $20, cpu_info_data /* data pointer to transfer */
|
||
|
+ li $21, 8 /* bytes left to transfer */
|
||
|
+ li $22, 1 /* set EP0 to TX state */
|
||
|
+ li $23, 0 /* NoData = 0 */
|
||
|
+
|
||
|
+ b _ep0_idle_state_fini
|
||
|
+ nop
|
||
|
+
|
||
|
+__ep0_set_data_address:
|
||
|
+ li $9, 0xffff0000
|
||
|
+ and $9, $25, $9
|
||
|
+ andi $8, $26, 0xffff
|
||
|
+ or $20, $9, $8 /* data address of next transfer */
|
||
|
+
|
||
|
+ b _ep0_idle_state_fini
|
||
|
+ nop
|
||
|
+
|
||
|
+__ep0_set_data_length:
|
||
|
+ li $9, 0xffff0000
|
||
|
+ and $9, $25, $9
|
||
|
+ andi $8, $26, 0xffff
|
||
|
+ or $21, $9, $8 /* data length of next transfer */
|
||
|
+
|
||
|
+ li $9, 0x48 /* SVDOUTPKTRDY and DATAEND */
|
||
|
+ sb $9, 0x12($27) /* CSR0 */
|
||
|
+
|
||
|
+ /* We must write packet to FIFO before EP1-IN interrupt here. */
|
||
|
+ b handle_epin1_intr
|
||
|
+ nop
|
||
|
+
|
||
|
+__ep0_flush_caches:
|
||
|
+ /* Flush dcache and invalidate icache. */
|
||
|
+ li $8, 0x80000000
|
||
|
+ addi $9, $8, 0x3fe0 /* total 16KB */
|
||
|
+
|
||
|
+1:
|
||
|
+ cache 0x0, 0($8) /* Index_Invalidate_I */
|
||
|
+ cache 0x1, 0($8) /* Index_Writeback_Inv_D */
|
||
|
+ bne $8, $9, 1b
|
||
|
+ addiu $8, $8, 32
|
||
|
+
|
||
|
+ /* flush write-buffer */
|
||
|
+ sync
|
||
|
+
|
||
|
+ /* Invalidate BTB */
|
||
|
+ mfc0 $8, $16, 7 /* CP0_CONFIG */
|
||
|
+ nop
|
||
|
+ ori $8, 2
|
||
|
+ mtc0 $8, $16, 7
|
||
|
+ nop
|
||
|
+
|
||
|
+ b _ep0_idle_state_fini
|
||
|
+ nop
|
||
|
+
|
||
|
+__ep0_prog_start1:
|
||
|
+ li $9, 0x48 /* SVDOUTPKTRDY and DATAEND */
|
||
|
+ sb $9, 0x12($27) /* CSR0 */
|
||
|
+
|
||
|
+ li $9, 0xffff0000
|
||
|
+ and $9, $25, $9
|
||
|
+ andi $8, $26, 0xffff
|
||
|
+ or $20, $9, $8 /* target address */
|
||
|
+
|
||
|
+ b xfer_d2i
|
||
|
+ li $19, 0x2000 /* 16KB data length */
|
||
|
+
|
||
|
+__ep0_prog_start2:
|
||
|
+ li $9, 0x48 /* SVDOUTPKTRDY and DATAEND */
|
||
|
+ sb $9, 0x12($27) /* CSR0 */
|
||
|
+
|
||
|
+ li $9, 0xffff0000
|
||
|
+ and $9, $25, $9
|
||
|
+ andi $8, $26, 0xffff
|
||
|
+ or $20, $9, $8 /* target address */
|
||
|
+
|
||
|
+ jalr $20 /* jump, and place the return address in $31 */
|
||
|
+ nop
|
||
|
+
|
||
|
+__ep0_prog_start2_return:
|
||
|
+/* User code can return to here after executing itself, by jumping to $31 */
|
||
|
+ b usb_boot_return
|
||
|
+ nop
|
||
|
+
|
||
|
+ /* 2.1.2 Standard setup request */
|
||
|
+_ep0_std_req:
|
||
|
+ andi $12, $25, 0xff00 /* check bRequest */
|
||
|
+ srl $12, $12, 8
|
||
|
+ sub $9, $12, 0x05 /* check USB_REQ_SET_ADDRESS */
|
||
|
+ bnez $9, __ep0_req_set_config
|
||
|
+ nop
|
||
|
+
|
||
|
+ /* Handle USB_REQ_SET_ADDRESS */
|
||
|
+__ep0_req_set_addr:
|
||
|
+ srl $9, $25, 16 /* get wValue */
|
||
|
+ sb $9, 0x0($27) /* set FADDR */
|
||
|
+ li $23, 1 /* NoData = 1 */
|
||
|
+ b _ep0_idle_state_fini
|
||
|
+ nop
|
||
|
+
|
||
|
+__ep0_req_set_config:
|
||
|
+ sub $9, $12, 0x09 /* check USB_REQ_SET_CONFIGURATION */
|
||
|
+ bnez $9, __ep0_req_get_desc
|
||
|
+ nop
|
||
|
+
|
||
|
+ /* Handle USB_REQ_SET_CONFIGURATION */
|
||
|
+ li $23, 1 /* NoData = 1 */
|
||
|
+ b _ep0_idle_state_fini
|
||
|
+ nop
|
||
|
+
|
||
|
+__ep0_req_get_desc:
|
||
|
+ sub $9, $12, 0x06 /* check USB_REQ_GET_DESCRIPTOR */
|
||
|
+ bnez $9, _ep0_idle_state_fini
|
||
|
+ li $23, 1 /* NoData = 1 */
|
||
|
+
|
||
|
+ /* Handle USB_REQ_GET_DESCRIPTOR */
|
||
|
+ li $23, 0 /* NoData = 0 */
|
||
|
+
|
||
|
+ srl $9, $25, 24 /* wValue >> 8 */
|
||
|
+ sub $8, $9, 0x01 /* check USB_DT_DEVICE */
|
||
|
+ beqz $8, ___ep0_get_dev_desc
|
||
|
+ srl $21, $26, 16 /* get wLength */
|
||
|
+ sub $8, $9, 0x02 /* check USB_DT_CONFIG */
|
||
|
+ beqz $8, ___ep0_get_conf_desc
|
||
|
+ sub $8, $9, 0x03 /* check USB_DT_STRING */
|
||
|
+ beqz $8, ___ep0_get_string_desc
|
||
|
+ sub $8, $9, 0x06 /* check USB_DT_DEVICE_QUALIFIER */
|
||
|
+ beqz $8, ___ep0_get_dev_qualifier
|
||
|
+ nop
|
||
|
+ b _ep0_idle_state_fini
|
||
|
+ nop
|
||
|
+
|
||
|
+___ep0_get_dev_desc:
|
||
|
+ load_addr $20, device_desc /* data pointer */
|
||
|
+ li $22, 1 /* set EP0 to TX state */
|
||
|
+ sub $8, $21, 18
|
||
|
+ blez $8, _ep0_idle_state_fini /* wLength <= 18 */
|
||
|
+ nop
|
||
|
+ li $21, 18 /* max length of device_desc */
|
||
|
+ b _ep0_idle_state_fini
|
||
|
+ nop
|
||
|
+
|
||
|
+___ep0_get_dev_qualifier:
|
||
|
+ load_addr $20, dev_qualifier /* data pointer */
|
||
|
+ li $22, 1 /* set EP0 to TX state */
|
||
|
+ sub $8, $21, 10
|
||
|
+ blez $8, _ep0_idle_state_fini /* wLength <= 10 */
|
||
|
+ nop
|
||
|
+ li $21, 10 /* max length of dev_qualifier */
|
||
|
+ b _ep0_idle_state_fini
|
||
|
+ nop
|
||
|
+
|
||
|
+___ep0_get_conf_desc:
|
||
|
+ load_addr $20, config_desc_fs /* data pointer of FS mode */
|
||
|
+ lbu $8, 0x01($27) /* read POWER */
|
||
|
+ andi $8, 0x10 /* test HS_MODE */
|
||
|
+ beqz $8, ___ep0_get_conf_desc2
|
||
|
+ nop
|
||
|
+ load_addr $20, config_desc_hs /* data pointer of HS mode */
|
||
|
+
|
||
|
+___ep0_get_conf_desc2:
|
||
|
+ li $22, 1 /* set EP0 to TX state */
|
||
|
+ sub $8, $21, 32
|
||
|
+ blez $8, _ep0_idle_state_fini /* wLength <= 32 */
|
||
|
+ nop
|
||
|
+ li $21, 32 /* max length of config_desc */
|
||
|
+ b _ep0_idle_state_fini
|
||
|
+ nop
|
||
|
+
|
||
|
+___ep0_get_string_desc:
|
||
|
+ li $22, 1 /* set EP0 to TX state */
|
||
|
+
|
||
|
+ srl $9, $25, 16 /* wValue & 0xff */
|
||
|
+ andi $9, 0xff
|
||
|
+
|
||
|
+ sub $8, $9, 1
|
||
|
+ beqz $8, ___ep0_get_string_manufacture
|
||
|
+ sub $8, $9, 2
|
||
|
+ beqz $8, ___ep0_get_string_product
|
||
|
+ nop
|
||
|
+
|
||
|
+___ep0_get_string_lang_ids:
|
||
|
+ load_addr $20, string_lang_ids /* data pointer */
|
||
|
+ b _ep0_idle_state_fini
|
||
|
+ li $21, 4 /* data length */
|
||
|
+
|
||
|
+___ep0_get_string_manufacture:
|
||
|
+ load_addr $20, string_manufacture /* data pointer */
|
||
|
+ b _ep0_idle_state_fini
|
||
|
+ li $21, 16 /* data length */
|
||
|
+
|
||
|
+___ep0_get_string_product:
|
||
|
+ load_addr $20, string_product /* data pointer */
|
||
|
+ b _ep0_idle_state_fini
|
||
|
+ li $21, 46 /* data length */
|
||
|
+
|
||
|
+_ep0_idle_state_fini:
|
||
|
+ li $9, 0x40 /* SVDOUTPKTRDY */
|
||
|
+ beqz $23, _ep0_idle_state_fini2
|
||
|
+ nop
|
||
|
+ ori $9, $9, 0x08 /* DATAEND */
|
||
|
+_ep0_idle_state_fini2:
|
||
|
+ sb $9, 0x12($27) /* CSR0 */
|
||
|
+ beqz $22, check_intr_ep1in
|
||
|
+ nop
|
||
|
+
|
||
|
+ /* 2.2 Handle EP0 TX state interrupt */
|
||
|
+ep0_tx_state:
|
||
|
+ sub $9, $22, 1
|
||
|
+ bnez $9, check_intr_ep1in
|
||
|
+ nop
|
||
|
+
|
||
|
+ sub $9, $21, 64 /* max packetsize */
|
||
|
+ blez $9, _ep0_tx_state2 /* data count <= 64 */
|
||
|
+ ori $19, $21, 0
|
||
|
+ li $19, 64
|
||
|
+
|
||
|
+_ep0_tx_state2:
|
||
|
+ beqz $19, _ep0_tx_state3 /* send ZLP */
|
||
|
+ ori $18, $19, 0 /* record bytes to be transferred */
|
||
|
+ sub $21, $21, $19 /* decrement data count */
|
||
|
+
|
||
|
+_ep0_fifo_write_loop:
|
||
|
+ lbu $9, 0($20) /* read data */
|
||
|
+ sb $9, 0x20($27) /* load FIFO */
|
||
|
+ sub $19, $19, 1 /* decrement counter */
|
||
|
+ bnez $19, _ep0_fifo_write_loop
|
||
|
+ addi $20, $20, 1 /* increment data pointer */
|
||
|
+
|
||
|
+ sub $9, $18, 64 /* max packetsize */
|
||
|
+ beqz $9, _ep0_tx_state4
|
||
|
+ nop
|
||
|
+
|
||
|
+_ep0_tx_state3:
|
||
|
+ /* transferred bytes < max packetsize */
|
||
|
+ li $9, 0x0a /* set INPKTRDY and DATAEND */
|
||
|
+ sb $9, 0x12($27) /* CSR0 */
|
||
|
+ li $22, 0 /* set EP0 to IDLE state */
|
||
|
+ b check_intr_ep1in
|
||
|
+ nop
|
||
|
+
|
||
|
+_ep0_tx_state4:
|
||
|
+ /* transferred bytes == max packetsize */
|
||
|
+ li $9, 0x02 /* set INPKTRDY */
|
||
|
+ sb $9, 0x12($27) /* CSR0 */
|
||
|
+ b check_intr_ep1in
|
||
|
+ nop
|
||
|
+
|
||
|
+ /* 3. Check and handle EP1 BULK-IN interrupt */
|
||
|
+check_intr_ep1in:
|
||
|
+ andi $9, $10, 0x2 /* check EP1 IN interrupt */
|
||
|
+ beqz $9, check_intr_ep1out
|
||
|
+ nop
|
||
|
+
|
||
|
+handle_epin1_intr:
|
||
|
+ li $9, 1
|
||
|
+ sb $9, 0x0e($27) /* set INDEX 1 */
|
||
|
+ lbu $9, 0x12($27) /* read INCSR */
|
||
|
+
|
||
|
+ andi $8, $9, 0x2 /* check INCSR_FFNOTEMPT */
|
||
|
+ bnez $8, _epin1_tx_state4
|
||
|
+ nop
|
||
|
+
|
||
|
+_epin1_write_fifo:
|
||
|
+ lhu $9, 0x10($27) /* get INMAXP */
|
||
|
+ sub $8, $21, $9
|
||
|
+ blez $8, _epin1_tx_state1 /* bytes left <= INMAXP */
|
||
|
+ ori $19, $21, 0
|
||
|
+ ori $19, $9, 0
|
||
|
+
|
||
|
+_epin1_tx_state1:
|
||
|
+ beqz $19, _epin1_tx_state4 /* No data */
|
||
|
+ nop
|
||
|
+
|
||
|
+ sub $21, $21, $19 /* decrement data count */
|
||
|
+
|
||
|
+ srl $5, $19, 2 /* # of word */
|
||
|
+ andi $6, $19, 0x3 /* # of byte */
|
||
|
+ beqz $5, _epin1_tx_state2
|
||
|
+ nop
|
||
|
+
|
||
|
+_epin1_fifo_write_word:
|
||
|
+ lw $9, 0($20) /* read data from source address */
|
||
|
+ sw $9, 0x24($27) /* write FIFO */
|
||
|
+ sub $5, $5, 1 /* decrement counter */
|
||
|
+ bnez $5, _epin1_fifo_write_word
|
||
|
+ addiu $20, $20, 4 /* increment dest address */
|
||
|
+
|
||
|
+_epin1_tx_state2:
|
||
|
+ beqz $6, _epin1_tx_state3
|
||
|
+ nop
|
||
|
+
|
||
|
+_epin1_fifo_write_byte:
|
||
|
+ lbu $9, 0($20) /* read data from source address */
|
||
|
+ sb $9, 0x24($27) /* write FIFO */
|
||
|
+ sub $6, $6, 1 /* decrement counter */
|
||
|
+ bnez $6, _epin1_fifo_write_byte
|
||
|
+ addiu $20, $20, 1 /* increment dest address */
|
||
|
+
|
||
|
+_epin1_tx_state3:
|
||
|
+ li $9, 0x1
|
||
|
+ sb $9, 0x12($27) /* INCSR, set INPKTRDY */
|
||
|
+
|
||
|
+_epin1_tx_state4:
|
||
|
+ /* 4. Check and handle EP1 BULK-OUT interrupt */
|
||
|
+check_intr_ep1out:
|
||
|
+ lhu $9, 0x04($27) /* read INTROUT */
|
||
|
+ andi $9, 0x2
|
||
|
+ beqz $9, check_status_next
|
||
|
+ nop
|
||
|
+
|
||
|
+handle_epout1_intr:
|
||
|
+ li $9, 1
|
||
|
+ sb $9, 0x0e($27) /* set INDEX 1 */
|
||
|
+
|
||
|
+ lbu $9, 0x16($27) /* read OUTCSR */
|
||
|
+ andi $9, 0x1 /* check OUTPKTRDY */
|
||
|
+ beqz $9, check_status_next
|
||
|
+ nop
|
||
|
+
|
||
|
+_epout1_read_fifo:
|
||
|
+ lhu $19, 0x18($27) /* read OUTCOUNT */
|
||
|
+ srl $5, $19, 2 /* # of word */
|
||
|
+ andi $6, $19, 0x3 /* # of byte */
|
||
|
+ beqz $5, _epout1_rx_state1
|
||
|
+ nop
|
||
|
+
|
||
|
+_epout1_fifo_read_word:
|
||
|
+ lw $9, 0x24($27) /* read FIFO */
|
||
|
+ sw $9, 0($20) /* store to dest address */
|
||
|
+ sub $5, $5, 1 /* decrement counter */
|
||
|
+ bnez $5, _epout1_fifo_read_word
|
||
|
+ addiu $20, $20, 4 /* increment dest address */
|
||
|
+
|
||
|
+_epout1_rx_state1:
|
||
|
+ beqz $6, _epout1_rx_state2
|
||
|
+ nop
|
||
|
+
|
||
|
+_epout1_fifo_read_byte:
|
||
|
+ lbu $9, 0x24($27) /* read FIFO */
|
||
|
+ sb $9, 0($20) /* store to dest address */
|
||
|
+ sub $6, $6, 1 /* decrement counter */
|
||
|
+ bnez $6, _epout1_fifo_read_byte
|
||
|
+ addiu $20, $20, 1 /* increment dest address */
|
||
|
+
|
||
|
+_epout1_rx_state2:
|
||
|
+ sb $0, 0x16($27) /* clear OUTPKTRDY */
|
||
|
+
|
||
|
+check_status_next:
|
||
|
+ b usb_command_loop
|
||
|
+ nop
|
||
|
+
|
||
|
+/* Device/Configuration/Interface/Endpoint/String Descriptors */
|
||
|
+
|
||
|
+ .align 2
|
||
|
+device_desc:
|
||
|
+ .byte 0x12 /* bLength */
|
||
|
+ .byte 0x01 /* bDescriptorType */
|
||
|
+ .byte 0x00 /* bcdUSB */
|
||
|
+ .byte 0x02 /* bcdUSB */
|
||
|
+ .byte 0x00 /* bDeviceClass */
|
||
|
+ .byte 0x00 /* bDeviceSubClass */
|
||
|
+ .byte 0x00 /* bDeviceProtocol */
|
||
|
+ .byte 0x40 /* bMaxPacketSize0 */
|
||
|
+ .byte 0x1a /* idVendor */
|
||
|
+ .byte 0x60 /* idVendor */
|
||
|
+ .byte 0x40 /* idProduct */
|
||
|
+ .byte 0x47 /* idProduct */
|
||
|
+ .byte 0x00 /* bcdDevice */
|
||
|
+ .byte 0x01 /* bcdDevice */
|
||
|
+ .byte 0x01 /* iManufacturer */
|
||
|
+ .byte 0x02 /* iProduct */
|
||
|
+ .byte 0x00 /* iSerialNumber */
|
||
|
+ .byte 0x01 /* bNumConfigurations */
|
||
|
+
|
||
|
+ .align 2
|
||
|
+dev_qualifier:
|
||
|
+ .byte 0x0a /* bLength */
|
||
|
+ .byte 0x06 /* bDescriptorType */
|
||
|
+ .byte 0x00 /* bcdUSB */
|
||
|
+ .byte 0x02 /* bcdUSB */
|
||
|
+ .byte 0x00 /* bDeviceClass */
|
||
|
+ .byte 0x00 /* bDeviceSubClass */
|
||
|
+ .byte 0x00 /* bDeviceProtocol */
|
||
|
+ .byte 0x40 /* bMaxPacketSize0 */
|
||
|
+ .byte 0x01 /* bNumConfigurations */
|
||
|
+ .byte 0x00 /* bRESERVED */
|
||
|
+
|
||
|
+ .align 2
|
||
|
+config_desc_hs:
|
||
|
+ .byte 0x09 /* bLength */
|
||
|
+ .byte 0x02 /* bDescriptorType */
|
||
|
+ .byte 0x20 /* wTotalLength */
|
||
|
+ .byte 0x00 /* wTotalLength */
|
||
|
+ .byte 0x01 /* bNumInterfaces */
|
||
|
+ .byte 0x01 /* bConfigurationValue */
|
||
|
+ .byte 0x00 /* iConfiguration */
|
||
|
+ .byte 0xc0 /* bmAttributes */
|
||
|
+ .byte 0x01 /* MaxPower */
|
||
|
+intf_desc_hs:
|
||
|
+ .byte 0x09 /* bLength */
|
||
|
+ .byte 0x04 /* bDescriptorType */
|
||
|
+ .byte 0x00 /* bInterfaceNumber */
|
||
|
+ .byte 0x00 /* bAlternateSetting */
|
||
|
+ .byte 0x02 /* bNumEndpoints */
|
||
|
+ .byte 0xff /* bInterfaceClass */
|
||
|
+ .byte 0x00 /* bInterfaceSubClass */
|
||
|
+ .byte 0x50 /* bInterfaceProtocol */
|
||
|
+ .byte 0x00 /* iInterface */
|
||
|
+ep1_desc_hs:
|
||
|
+ .byte 0x07 /* bLength */
|
||
|
+ .byte 0x05 /* bDescriptorType */
|
||
|
+ .byte 0x01 /* bEndpointAddress */
|
||
|
+ .byte 0x02 /* bmAttributes */
|
||
|
+ .byte 0x00 /* wMaxPacketSize */
|
||
|
+ .byte 0x02 /* wMaxPacketSize */
|
||
|
+ .byte 0x00 /* bInterval */
|
||
|
+ep2_desc_hs:
|
||
|
+ .byte 0x07 /* bLength */
|
||
|
+ .byte 0x05 /* bDescriptorType */
|
||
|
+ .byte 0x81 /* bEndpointAddress */
|
||
|
+ .byte 0x02 /* bmAttributes */
|
||
|
+ .byte 0x00 /* wMaxPacketSize */
|
||
|
+ .byte 0x02 /* wMaxPacketSize */
|
||
|
+ .byte 0x00 /* bInterval */
|
||
|
+
|
||
|
+ .align 2
|
||
|
+config_desc_fs:
|
||
|
+ .byte 0x09 /* bLength */
|
||
|
+ .byte 0x02 /* bDescriptorType */
|
||
|
+ .byte 0x20 /* wTotalLength */
|
||
|
+ .byte 0x00 /* wTotalLength */
|
||
|
+ .byte 0x01 /* bNumInterfaces */
|
||
|
+ .byte 0x01 /* bConfigurationValue */
|
||
|
+ .byte 0x00 /* iConfiguration */
|
||
|
+ .byte 0xc0 /* bmAttributes */
|
||
|
+ .byte 0x01 /* MaxPower */
|
||
|
+intf_desc_fs:
|
||
|
+ .byte 0x09 /* bLength */
|
||
|
+ .byte 0x04 /* bDescriptorType */
|
||
|
+ .byte 0x00 /* bInterfaceNumber */
|
||
|
+ .byte 0x00 /* bAlternateSetting */
|
||
|
+ .byte 0x02 /* bNumEndpoints */
|
||
|
+ .byte 0xff /* bInterfaceClass */
|
||
|
+ .byte 0x00 /* bInterfaceSubClass */
|
||
|
+ .byte 0x50 /* bInterfaceProtocol */
|
||
|
+ .byte 0x00 /* iInterface */
|
||
|
+ep1_desc_fs:
|
||
|
+ .byte 0x07 /* bLength */
|
||
|
+ .byte 0x05 /* bDescriptorType */
|
||
|
+ .byte 0x01 /* bEndpointAddress */
|
||
|
+ .byte 0x02 /* bmAttributes */
|
||
|
+ .byte 0x40 /* wMaxPacketSize */
|
||
|
+ .byte 0x00 /* wMaxPacketSize */
|
||
|
+ .byte 0x00 /* bInterval */
|
||
|
+ep2_desc_fs:
|
||
|
+ .byte 0x07 /* bLength */
|
||
|
+ .byte 0x05 /* bDescriptorType */
|
||
|
+ .byte 0x81 /* bEndpointAddress */
|
||
|
+ .byte 0x02 /* bmAttributes */
|
||
|
+ .byte 0x40 /* wMaxPacketSize */
|
||
|
+ .byte 0x00 /* wMaxPacketSize */
|
||
|
+ .byte 0x00 /* bInterval */
|
||
|
+
|
||
|
+ .align 2
|
||
|
+string_lang_ids:
|
||
|
+ .byte 0x04
|
||
|
+ .byte 0x03
|
||
|
+ .byte 0x09
|
||
|
+ .byte 0x04
|
||
|
+
|
||
|
+ .align 2
|
||
|
+string_manufacture:
|
||
|
+ .byte 0x10
|
||
|
+ .byte 0x03
|
||
|
+ .byte 0x49
|
||
|
+ .byte 0x00
|
||
|
+ .byte 0x6e
|
||
|
+ .byte 0x00
|
||
|
+ .byte 0x67
|
||
|
+ .byte 0x00
|
||
|
+ .byte 0x65
|
||
|
+ .byte 0x00
|
||
|
+ .byte 0x6e
|
||
|
+ .byte 0x00
|
||
|
+ .byte 0x69
|
||
|
+ .byte 0x00
|
||
|
+ .byte 0x63
|
||
|
+ .byte 0x00
|
||
|
+
|
||
|
+ .align 2
|
||
|
+string_product:
|
||
|
+ .byte 0x2e
|
||
|
+ .byte 0x03
|
||
|
+ .byte 0x4a
|
||
|
+ .byte 0x00
|
||
|
+ .byte 0x5a
|
||
|
+ .byte 0x00
|
||
|
+ .byte 0x34
|
||
|
+ .byte 0x00
|
||
|
+ .byte 0x37
|
||
|
+ .byte 0x00
|
||
|
+ .byte 0x34
|
||
|
+ .byte 0x00
|
||
|
+ .byte 0x30
|
||
|
+ .byte 0x00
|
||
|
+ .byte 0x20
|
||
|
+ .byte 0x00
|
||
|
+ .byte 0x55
|
||
|
+ .byte 0x00
|
||
|
+ .byte 0x53
|
||
|
+ .byte 0x00
|
||
|
+ .byte 0x42
|
||
|
+ .byte 0x00
|
||
|
+ .byte 0x20
|
||
|
+ .byte 0x00
|
||
|
+ .byte 0x42
|
||
|
+ .byte 0x00
|
||
|
+ .byte 0x6f
|
||
|
+ .byte 0x00
|
||
|
+ .byte 0x6f
|
||
|
+ .byte 0x00
|
||
|
+ .byte 0x74
|
||
|
+ .byte 0x00
|
||
|
+ .byte 0x20
|
||
|
+ .byte 0x00
|
||
|
+ .byte 0x44
|
||
|
+ .byte 0x00
|
||
|
+ .byte 0x65
|
||
|
+ .byte 0x00
|
||
|
+ .byte 0x76
|
||
|
+ .byte 0x00
|
||
|
+ .byte 0x69
|
||
|
+ .byte 0x00
|
||
|
+ .byte 0x63
|
||
|
+ .byte 0x00
|
||
|
+ .byte 0x65
|
||
|
+ .byte 0x00
|
||
|
+
|
||
|
+ .align 2
|
||
|
+cpu_info_data:
|
||
|
+ .byte 0x4a
|
||
|
+ .byte 0x5a
|
||
|
+ .byte 0x34
|
||
|
+ .byte 0x37
|
||
|
+ .byte 0x34
|
||
|
+ .byte 0x30
|
||
|
+ .byte 0x56
|
||
|
+ .byte 0x31
|
||
|
+usbboot_end:
|
||
|
+
|
||
|
+ .set reorder
|
||
|
--
|
||
|
1.7.9.5
|
||
|
|