/////////////////////////////////////////////////////////////////////// // spi.c // // Written: Jaocb Pease jacob.pease@okstate.edu 7/22/2024 // // Purpose: SPI Controller API for bootloader // // // // A component of the Wally configurable RISC-V project. // // Copyright (C) 2021-23 Harvey Mudd College & Oklahoma State University // // SPDX-License-Identifier: Apache-2.0 WITH SHL-2.1 // // Licensed under the Solderpad Hardware License v 2.1 (the // “License”); you may not use this file except in compliance with the // License, or, at your option, the Apache License version 2.0. You // may obtain a copy of the License at // // https://solderpad.org/licenses/SHL-2.1/ // // Unless required by applicable law or agreed to in writing, any work // distributed under the License is distributed on an “AS IS” BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or // implied. See the License for the specific language governing // permissions and limitations under the License. /////////////////////////////////////////////////////////////////////// #include "spi.h" // Write to a register /* inline void write_reg(uintptr_t addr, uint32_t value) { */ /* volatile uint32_t * loc = (volatile uint32_t *) addr; */ /* *loc = value; */ /* } */ /* // Read a register */ /* inline uint32_t read_reg(uintptr_t addr) { */ /* return *(volatile uint32_t *) addr; */ /* } */ /* // Queues a single byte in the transfer fifo */ /* inline void spi_sendbyte(uint8_t byte) { */ /* // Write byte to transfer fifo */ /* write_reg(SPI_TXDATA, byte); */ /* } */ /* inline void waittx() { */ /* while(!(read_reg(SPI_IP) & 1)) {} */ /* } */ /* inline void waitrx() { */ /* while(read_reg(SPI_IP) & 2) {} */ /* } */ uint8_t spi_txrx(uint8_t byte) { spi_sendbyte(0xFF); waittx(); return spi_readbyte(); } /* inline uint8_t spi_readbyte() { */ /* return read_reg(SPI_RXDATA); */ /* } */ uint64_t spi_read64() { uint64_t r; uint8_t rbyte; int i; for (i = 0; i < 8; i++) { spi_sendbyte(0xFF); } waittx(); for (i = 0; i < 8; i++) { rbyte = spi_readbyte(); r = r | (rbyte << ((8 - 1 - i)*8)); } return r; } void spi_set_clock(uint32_t clkin, uint32_t clkout) { uint32_t div = (clkin/(2*clkout)) - 1; write_reg(SPI_SCKDIV, div); } // Initialize Sifive FU540 based SPI Controller void spi_init(uint32_t clkin) { // Enable interrupts write_reg(SPI_IE, 0x3); // Set TXMARK to 1. If the number of entries is < 1 // IP's txwm field will go high. // Set RXMARK to 0. If the number of entries is > 0 // IP's rwxm field will go high. write_reg(SPI_TXMARK, 1); write_reg(SPI_RXMARK, 0); // Set Delay 0 to default write_reg(SPI_DELAY0, SIFIVE_SPI_DELAY0_CSSCK(1) | SIFIVE_SPI_DELAY0_SCKCS(1)); // Set Delay 1 to default write_reg(SPI_DELAY1, SIFIVE_SPI_DELAY1_INTERCS(1) | SIFIVE_SPI_DELAY1_INTERXFR(0)); // Initialize the SPI controller clock to // div = (20MHz/(2*400kHz)) - 1 = 24 = 0x18 write_reg(SPI_SCKDIV, 0x18); }