| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152 | #include "scsi_accel_asm.h"#include "BlueSCSI_platform.h"// Optimized ASM blocks for the SCSI communication subroutine// Take 8 bits from d and format them for writing// d is name of data operand, b is bit offset, x is unique label#define ASM_LOAD_DATA(d, b, x) \"    load_data1_" x "_%=: \n" \"        ubfx    %[tmp1], %[" d "], #" b ", #8 \n" \"        ldr     %[tmp1], [%[byte_lookup], %[tmp1], lsl #2] \n"// Write data to SCSI port and set REQ high#define ASM_SEND_DATA(x) \"    send_data" x "_%=: \n" \"        str     %[tmp1], [%[out_port_bop]] \n"// Read data from SCSI port, set REQ high, store data to d at bit offset b#define ASM_RECV_DATA(d, b, x) \"    recv_data" x "_%=: \n" \"        ldr     %[tmp1], [%[in_port_istat]] \n" \"        mov     %[tmp2], %[req_high_bop] \n" \"        str     %[tmp2], [%[out_port_bop]] \n" \"        ubfx    %[tmp1], %[tmp1], %[data_in_shift], #8 \n" \"        bfi     %[" d "], %[tmp1], #" b ", #8 \n"// Wait for ACK to be high, set REQ low, wait ACK low#define ASM_HANDSHAKE(x) \"        ldr     %[tmp2], [%[ack_pin_bb]] \n" \"        str     %[tmp2], [%[req_pin_bb]] \n" \"        cbnz    %[tmp2], req_is_low_now" x "_%= \n" \"        ldr     %[tmp2], [%[ack_pin_bb]] \n" \"        str     %[tmp2], [%[req_pin_bb]] \n" \"        cbnz    %[tmp2], req_is_low_now" x "_%= \n" \"        ldr     %[tmp2], [%[ack_pin_bb]] \n" \"        str     %[tmp2], [%[req_pin_bb]] \n" \"        cbnz    %[tmp2], req_is_low_now" x "_%= \n" \"    wait_ack_inactive" x "_%=: \n" \"        ldr     %[tmp2], [%[ack_pin_bb]] \n" \"        str     %[tmp2], [%[req_pin_bb]] \n" \"        cbnz    %[tmp2], req_is_low_now" x "_%= \n" \"        ldr     %[tmp2], [%[reset_flag]] \n" \"        cbnz    %[tmp2], req_is_low_now" x "_%= \n" \"        b.n     wait_ack_inactive" x "_%= \n" \"    req_is_low_now" x "_%=: \n" \"        ldr     %[tmp2], [%[ack_pin_bb]] \n" \"        cbz     %[tmp2], over_ack_active" x "_%= \n" \"        ldr     %[tmp2], [%[ack_pin_bb]] \n" \"        cbz     %[tmp2], over_ack_active" x "_%= \n" \"        ldr     %[tmp2], [%[ack_pin_bb]] \n" \"        cbz     %[tmp2], over_ack_active" x "_%= \n" \"        ldr     %[tmp2], [%[ack_pin_bb]] \n" \"        cbz     %[tmp2], over_ack_active" x "_%= \n" \"    wait_ack_active" x "_%=: \n" \"        ldr     %[tmp2], [%[ack_pin_bb]] \n" \"        cbz     %[tmp2], over_ack_active" x "_%= \n" \"        ldr     %[tmp2], [%[reset_flag]] \n" \"        cbnz    %[tmp2], over_ack_active" x "_%= \n" \"        b.n     wait_ack_active" x "_%= \n" \"    over_ack_active" x "_%=: \n" \// Send bytes to SCSI bus using the asynchronous handshake mechanism// Takes 4 bytes at a time for sending from buf.void scsi_accel_asm_send(const uint32_t *buf, uint32_t num_words, volatile int *resetFlag){    volatile uint32_t *out_port_bop = (volatile uint32_t*)&GPIO_BOP(SCSI_OUT_PORT);    const uint32_t *byte_lookup = g_scsi_out_byte_to_bop;    uint32_t ack_pin_bb = PERIPH_BB_BASE + (((uint32_t)&GPIO_ISTAT(SCSI_ACK_PORT)) - APB1_BUS_BASE) * 32 + SCSI_IN_ACK_IDX * 4;    uint32_t req_pin_bb = PERIPH_BB_BASE + (((uint32_t)out_port_bop) - APB1_BUS_BASE) * 32 + (SCSI_OUT_REQ_IDX + 16) * 4;    register uint32_t tmp1 = 0;    register uint32_t tmp2 = 0;    register uint32_t data = 0;    asm volatile (    "   ldr      %[data], [%[buf]], #4 \n" \        ASM_LOAD_DATA("data", "0", "first")    "inner_loop_%=: \n" \        ASM_SEND_DATA("0")        ASM_LOAD_DATA("data", "8", "8")        ASM_HANDSHAKE("0")        ASM_SEND_DATA("8")        ASM_LOAD_DATA("data", "16", "16")        ASM_HANDSHAKE("8")        ASM_SEND_DATA("16")        ASM_LOAD_DATA("data", "24", "24")        ASM_HANDSHAKE("16")        ASM_SEND_DATA("24")    "   ldr      %[data], [%[buf]], #4 \n" \        ASM_LOAD_DATA("data", "0", "0")        ASM_HANDSHAKE("24")    "   subs     %[num_words], %[num_words], #1 \n" \    "   bne     inner_loop_%= \n"    : /* Output */ [tmp1] "+l" (tmp1), [tmp2] "+l" (tmp2), [data] "+r" (data),                   [buf] "+r" (buf), [num_words] "+r" (num_words)    : /* Input */ [ack_pin_bb] "r" (ack_pin_bb),                  [req_pin_bb] "r" (req_pin_bb),                  [out_port_bop] "r"(out_port_bop),                  [byte_lookup] "r" (byte_lookup),                  [reset_flag] "r" (resetFlag)    : /* Clobber */ );    SCSI_RELEASE_DATA_REQ();}// Read bytes from SCSI bus using asynchronous handshake mechanism// Takes 4 bytes at a time.void scsi_accel_asm_recv(uint32_t *buf, uint32_t num_words, volatile int *resetFlag){    volatile uint32_t *out_port_bop = (volatile uint32_t*)&GPIO_BOP(SCSI_OUT_PORT);    volatile uint32_t *in_port_istat = (volatile uint32_t*)&GPIO_ISTAT(SCSI_IN_PORT);    uint32_t ack_pin_bb = PERIPH_BB_BASE + (((uint32_t)&GPIO_ISTAT(SCSI_ACK_PORT)) - APB1_BUS_BASE) * 32 + SCSI_IN_ACK_IDX * 4;    uint32_t req_pin_bb = PERIPH_BB_BASE + (((uint32_t)out_port_bop) - APB1_BUS_BASE) * 32 + (SCSI_OUT_REQ_IDX + 16) * 4;    register uint32_t tmp1 = 0;    register uint32_t tmp2 = 0;    register uint32_t data = 0;    asm volatile (    "inner_loop_%=: \n" \        ASM_HANDSHAKE("0")        ASM_RECV_DATA("data", "0", "0")        ASM_HANDSHAKE("8")        ASM_RECV_DATA("data", "8", "8")        ASM_HANDSHAKE("16")        ASM_RECV_DATA("data", "16", "16")        ASM_HANDSHAKE("24")        ASM_RECV_DATA("data", "24", "24")    "   mvn      %[data], %[data] \n" \    "   str      %[data], [%[buf]], #4 \n" \    "   subs     %[num_words], %[num_words], #1 \n" \    "   bne     inner_loop_%= \n"    : /* Output */ [tmp1] "+l" (tmp1), [tmp2] "+l" (tmp2), [data] "+r" (data),                   [buf] "+r" (buf), [num_words] "+r" (num_words)    : /* Input */ [ack_pin_bb] "r" (ack_pin_bb),                  [req_pin_bb] "r" (req_pin_bb),                  [out_port_bop] "r"(out_port_bop),                  [in_port_istat] "r" (in_port_istat),                  [reset_flag] "r" (resetFlag),                  [data_in_shift] "I" (SCSI_IN_SHIFT),                  [req_high_bop] "I" (SCSI_OUT_REQ)    : /* Clobber */ );    SCSI_RELEASE_DATA_REQ();}
 |