mirror of
https://github.com/nippur72/Apple1_MiST.git
synced 2025-02-28 02:30:07 +00:00
260 lines
6.5 KiB
Verilog
260 lines
6.5 KiB
Verilog
//
|
|
// data_io.v
|
|
//
|
|
// data_io for the MiST board
|
|
// http://code.google.com/p/mist-board/
|
|
//
|
|
// Copyright (c) 2014 Till Harbaum <till@harbaum.org>
|
|
//
|
|
// This source file 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 3 of the License, or
|
|
// (at your option) any later version.
|
|
//
|
|
// This source file 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, see <http://www.gnu.org/licenses/>.
|
|
//
|
|
///////////////////////////////////////////////////////////////////////
|
|
|
|
module data_io
|
|
(
|
|
input clk_sys,
|
|
input SPI_SCK,
|
|
input SPI_SS2,
|
|
input SPI_SS4,
|
|
input SPI_DI,
|
|
inout SPI_DO,
|
|
|
|
input clkref_n, // assert ioctl_wr one cycle after clkref stobe (negative active)
|
|
|
|
// ARM -> FPGA download
|
|
output reg ioctl_download = 0, // signal indicating an active download
|
|
output reg ioctl_upload = 0, // signal indicating an active upload
|
|
output reg [7:0] ioctl_index, // menu index used to upload the file ([7:6] - extension index, [5:0] - menu index)
|
|
// Note: this is also set for user_io mounts.
|
|
// Valid when ioctl_download = 1 or when img_mounted strobe is active in user_io.
|
|
output reg ioctl_wr, // strobe indicating ioctl_dout valid
|
|
output reg [24:0] ioctl_addr,
|
|
output reg [7:0] ioctl_dout,
|
|
input [7:0] ioctl_din,
|
|
output reg [23:0] ioctl_fileext, // file extension
|
|
output reg [31:0] ioctl_filesize // file size
|
|
);
|
|
|
|
parameter START_ADDR = 25'd0;
|
|
parameter ROM_DIRECT_UPLOAD = 0;
|
|
|
|
/////////////////////////////// DOWNLOADING ///////////////////////////////
|
|
|
|
reg [7:0] data_w;
|
|
reg [7:0] data_w2 = 0;
|
|
reg [3:0] cnt;
|
|
reg rclk = 0;
|
|
reg rclk2 = 0;
|
|
reg addr_reset = 0;
|
|
reg downloading_reg = 0;
|
|
reg uploading_reg = 0;
|
|
reg reg_do;
|
|
|
|
localparam DIO_FILE_TX = 8'h53;
|
|
localparam DIO_FILE_TX_DAT = 8'h54;
|
|
localparam DIO_FILE_INDEX = 8'h55;
|
|
localparam DIO_FILE_INFO = 8'h56;
|
|
localparam DIO_FILE_RX = 8'h57;
|
|
localparam DIO_FILE_RX_DAT = 8'h58;
|
|
|
|
assign SPI_DO = reg_do;
|
|
|
|
// data_io has its own SPI interface to the io controller
|
|
always@(negedge SPI_SCK or posedge SPI_SS2) begin : SPI_TRANSMITTER
|
|
reg [7:0] dout_r;
|
|
|
|
if(SPI_SS2) begin
|
|
reg_do <= 1'bZ;
|
|
end else begin
|
|
if (cnt == 15) dout_r <= ioctl_din;
|
|
reg_do <= dout_r[~cnt[2:0]];
|
|
end
|
|
end
|
|
|
|
|
|
always@(posedge SPI_SCK, posedge SPI_SS2) begin : SPI_RECEIVER
|
|
reg [6:0] sbuf;
|
|
reg [24:0] addr;
|
|
reg [7:0] cmd;
|
|
reg [5:0] bytecnt;
|
|
|
|
if(SPI_SS2) begin
|
|
bytecnt <= 0;
|
|
cnt <= 0;
|
|
end else begin
|
|
// don't shift in last bit. It is evaluated directly
|
|
// when writing to ram
|
|
if(cnt != 15) sbuf <= { sbuf[5:0], SPI_DI};
|
|
|
|
// count 0-7 8-15 8-15 ...
|
|
if(cnt != 15) cnt <= cnt + 1'd1;
|
|
else cnt <= 8;
|
|
|
|
// finished command byte
|
|
if(cnt == 7) cmd <= {sbuf, SPI_DI};
|
|
|
|
if(cnt == 15) begin
|
|
case (cmd)
|
|
// prepare/end transmission
|
|
DIO_FILE_TX: begin
|
|
// prepare
|
|
if(SPI_DI) begin
|
|
addr_reset <= ~addr_reset;
|
|
downloading_reg <= 1;
|
|
end else begin
|
|
downloading_reg <= 0;
|
|
end
|
|
end
|
|
|
|
DIO_FILE_RX: begin
|
|
// prepare
|
|
if(SPI_DI) begin
|
|
addr_reset <= ~addr_reset;
|
|
uploading_reg <= 1;
|
|
end else begin
|
|
uploading_reg <= 0;
|
|
end
|
|
end
|
|
|
|
// command 0x57: DIO_FILE_RX_DAT
|
|
// command 0x54: DIO_FILE_TX_DAT
|
|
DIO_FILE_RX_DAT,
|
|
DIO_FILE_TX_DAT: begin
|
|
data_w <= {sbuf, SPI_DI};
|
|
rclk <= ~rclk;
|
|
end
|
|
|
|
// expose file (menu) index
|
|
DIO_FILE_INDEX: ioctl_index <= {sbuf, SPI_DI};
|
|
|
|
// receiving FAT directory entry (mist-firmware/fat.h - DIRENTRY)
|
|
DIO_FILE_INFO: begin
|
|
bytecnt <= bytecnt + 1'd1;
|
|
case (bytecnt)
|
|
8'h08: ioctl_fileext[23:16] <= {sbuf, SPI_DI};
|
|
8'h09: ioctl_fileext[15: 8] <= {sbuf, SPI_DI};
|
|
8'h0A: ioctl_fileext[ 7: 0] <= {sbuf, SPI_DI};
|
|
8'h1C: ioctl_filesize[ 7: 0] <= {sbuf, SPI_DI};
|
|
8'h1D: ioctl_filesize[15: 8] <= {sbuf, SPI_DI};
|
|
8'h1E: ioctl_filesize[23:16] <= {sbuf, SPI_DI};
|
|
8'h1F: ioctl_filesize[31:24] <= {sbuf, SPI_DI};
|
|
endcase
|
|
end
|
|
endcase
|
|
end
|
|
end
|
|
end
|
|
|
|
// direct SD Card->FPGA transfer
|
|
generate if (ROM_DIRECT_UPLOAD == 1) begin
|
|
|
|
always@(posedge SPI_SCK, posedge SPI_SS4) begin : SPI_DIRECT_RECEIVER
|
|
reg [6:0] sbuf2;
|
|
reg [2:0] cnt2;
|
|
reg [9:0] bytecnt;
|
|
|
|
if(SPI_SS4) begin
|
|
cnt2 <= 0;
|
|
bytecnt <= 0;
|
|
end else begin
|
|
// don't shift in last bit. It is evaluated directly
|
|
// when writing to ram
|
|
if(cnt2 != 7)
|
|
sbuf2 <= { sbuf2[5:0], SPI_DO };
|
|
|
|
cnt2 <= cnt2 + 1'd1;
|
|
|
|
// received a byte
|
|
if(cnt2 == 7) begin
|
|
bytecnt <= bytecnt + 1'd1;
|
|
// read 514 byte/sector (512 + 2 CRC)
|
|
if (bytecnt == 513) bytecnt <= 0;
|
|
// don't send the CRC bytes
|
|
if (~bytecnt[9]) begin
|
|
data_w2 <= {sbuf2, SPI_DO};
|
|
rclk2 <= ~rclk2;
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
end
|
|
endgenerate
|
|
|
|
always@(posedge clk_sys) begin : DATA_OUT
|
|
// synchronisers
|
|
reg rclkD, rclkD2;
|
|
reg rclk2D, rclk2D2;
|
|
reg addr_resetD, addr_resetD2;
|
|
|
|
reg wr_int, wr_int_direct, rd_int;
|
|
reg [24:0] addr;
|
|
reg [31:0] filepos;
|
|
|
|
// bring flags from spi clock domain into core clock domain
|
|
{ rclkD, rclkD2 } <= { rclk, rclkD };
|
|
{ rclk2D ,rclk2D2 } <= { rclk2, rclk2D };
|
|
{ addr_resetD, addr_resetD2 } <= { addr_reset, addr_resetD };
|
|
|
|
ioctl_wr <= 0;
|
|
|
|
if (!downloading_reg) begin
|
|
ioctl_download <= 0;
|
|
wr_int <= 0;
|
|
wr_int_direct <= 0;
|
|
end
|
|
|
|
if (!uploading_reg) begin
|
|
ioctl_upload <= 0;
|
|
rd_int <= 0;
|
|
end
|
|
|
|
if (~clkref_n) begin
|
|
rd_int <= 0;
|
|
wr_int <= 0;
|
|
wr_int_direct <= 0;
|
|
if (wr_int || wr_int_direct) begin
|
|
ioctl_dout <= wr_int ? data_w : data_w2;
|
|
ioctl_wr <= 1;
|
|
addr <= addr + 1'd1;
|
|
ioctl_addr <= addr;
|
|
end
|
|
if (rd_int) begin
|
|
ioctl_addr <= ioctl_addr + 1'd1;
|
|
end
|
|
end
|
|
|
|
// detect transfer start from the SPI receiver
|
|
if(addr_resetD ^ addr_resetD2) begin
|
|
addr <= START_ADDR;
|
|
ioctl_addr <= START_ADDR;
|
|
filepos <= 0;
|
|
ioctl_download <= downloading_reg;
|
|
ioctl_upload <= uploading_reg;
|
|
end
|
|
|
|
// detect new byte from the SPI receiver
|
|
if (rclkD ^ rclkD2) begin
|
|
wr_int <= downloading_reg;
|
|
rd_int <= uploading_reg;
|
|
end
|
|
// direct transfer receiver
|
|
if (rclk2D ^ rclk2D2 && filepos != ioctl_filesize) begin
|
|
filepos <= filepos + 1'd1;
|
|
wr_int_direct <= 1;
|
|
end
|
|
end
|
|
|
|
endmodule
|