mirror of
https://github.com/openwrt/openwrt.git
synced 2025-01-22 04:18:10 +00:00
1171 lines
34 KiB
C
1171 lines
34 KiB
C
|
/*
|
||
|
* (C) Copyright 2005
|
||
|
* Oxford Semiconductor Ltd
|
||
|
*
|
||
|
* 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
|
||
|
*/
|
||
|
#include <common.h>
|
||
|
#include <asm/arch/clock.h>
|
||
|
|
||
|
/**
|
||
|
* SATA related definitions
|
||
|
*/
|
||
|
#define ATA_PORT_CTL 0
|
||
|
#define ATA_PORT_FEATURE 1
|
||
|
#define ATA_PORT_NSECT 2
|
||
|
#define ATA_PORT_LBAL 3
|
||
|
#define ATA_PORT_LBAM 4
|
||
|
#define ATA_PORT_LBAH 5
|
||
|
#define ATA_PORT_DEVICE 6
|
||
|
#define ATA_PORT_COMMAND 7
|
||
|
|
||
|
/* The offsets to the SATA registers */
|
||
|
#define SATA_ORB1_OFF 0
|
||
|
#define SATA_ORB2_OFF 1
|
||
|
#define SATA_ORB3_OFF 2
|
||
|
#define SATA_ORB4_OFF 3
|
||
|
#define SATA_ORB5_OFF 4
|
||
|
|
||
|
#define SATA_FIS_ACCESS 11
|
||
|
#define SATA_INT_STATUS_OFF 12 /* Read only */
|
||
|
#define SATA_INT_CLR_OFF 12 /* Write only */
|
||
|
#define SATA_INT_ENABLE_OFF 13 /* Read only */
|
||
|
#define SATA_INT_ENABLE_SET_OFF 13 /* Write only */
|
||
|
#define SATA_INT_ENABLE_CLR_OFF 14 /* Write only */
|
||
|
#define SATA_VERSION_OFF 15
|
||
|
#define SATA_CONTROL_OFF 23
|
||
|
#define SATA_COMMAND_OFF 24
|
||
|
#define SATA_PORT_CONTROL_OFF 25
|
||
|
#define SATA_DRIVE_CONTROL_OFF 26
|
||
|
|
||
|
/* The offsets to the link registers that are access in an asynchronous manner */
|
||
|
#define SATA_LINK_DATA 28
|
||
|
#define SATA_LINK_RD_ADDR 29
|
||
|
#define SATA_LINK_WR_ADDR 30
|
||
|
#define SATA_LINK_CONTROL 31
|
||
|
|
||
|
/* SATA interrupt status register fields */
|
||
|
#define SATA_INT_STATUS_EOC_RAW_BIT ( 0 + 16)
|
||
|
#define SATA_INT_STATUS_ERROR_BIT ( 2 + 16)
|
||
|
#define SATA_INT_STATUS_EOADT_RAW_BIT ( 1 + 16)
|
||
|
|
||
|
/* SATA core command register commands */
|
||
|
#define SATA_CMD_WRITE_TO_ORB_REGS 2
|
||
|
#define SATA_CMD_WRITE_TO_ORB_REGS_NO_COMMAND 4
|
||
|
|
||
|
#define SATA_CMD_BUSY_BIT 7
|
||
|
|
||
|
#define SATA_SCTL_CLR_ERR 0x00000316UL
|
||
|
|
||
|
#define SATA_LBAL_BIT 0
|
||
|
#define SATA_LBAM_BIT 8
|
||
|
#define SATA_LBAH_BIT 16
|
||
|
#define SATA_HOB_LBAH_BIT 24
|
||
|
#define SATA_DEVICE_BIT 24
|
||
|
#define SATA_NSECT_BIT 0
|
||
|
#define SATA_HOB_NSECT_BIT 8
|
||
|
#define SATA_LBA32_BIT 0
|
||
|
#define SATA_LBA40_BIT 8
|
||
|
#define SATA_FEATURE_BIT 16
|
||
|
#define SATA_COMMAND_BIT 24
|
||
|
#define SATA_CTL_BIT 24
|
||
|
|
||
|
/* ATA status (7) register field definitions */
|
||
|
#define ATA_STATUS_BSY_BIT 7
|
||
|
#define ATA_STATUS_DRDY_BIT 6
|
||
|
#define ATA_STATUS_DF_BIT 5
|
||
|
#define ATA_STATUS_DRQ_BIT 3
|
||
|
#define ATA_STATUS_ERR_BIT 0
|
||
|
|
||
|
/* ATA device (6) register field definitions */
|
||
|
#define ATA_DEVICE_FIXED_MASK 0xA0
|
||
|
#define ATA_DEVICE_DRV_BIT 4
|
||
|
#define ATA_DEVICE_DRV_NUM_BITS 1
|
||
|
#define ATA_DEVICE_LBA_BIT 6
|
||
|
|
||
|
/* ATA Command register initiated commands */
|
||
|
#define ATA_CMD_INIT 0x91
|
||
|
#define ATA_CMD_IDENT 0xEC
|
||
|
|
||
|
#define SATA_STD_ASYNC_REGS_OFF 0x20
|
||
|
#define SATA_SCR_STATUS 0
|
||
|
#define SATA_SCR_ERROR 1
|
||
|
#define SATA_SCR_CONTROL 2
|
||
|
#define SATA_SCR_ACTIVE 3
|
||
|
#define SATA_SCR_NOTIFICAION 4
|
||
|
|
||
|
#define SATA_BURST_BUF_FORCE_EOT_BIT 0
|
||
|
#define SATA_BURST_BUF_DATA_INJ_ENABLE_BIT 1
|
||
|
#define SATA_BURST_BUF_DIR_BIT 2
|
||
|
#define SATA_BURST_BUF_DATA_INJ_END_BIT 3
|
||
|
#define SATA_BURST_BUF_FIFO_DIS_BIT 4
|
||
|
#define SATA_BURST_BUF_DIS_DREQ_BIT 5
|
||
|
#define SATA_BURST_BUF_DREQ_BIT 6
|
||
|
|
||
|
#define SATA_OPCODE_MASK 0x3
|
||
|
|
||
|
#define SATA_DMA_CHANNEL 0
|
||
|
|
||
|
#define DMA_CTRL_STATUS (0x0)
|
||
|
#define DMA_BASE_SRC_ADR (0x4)
|
||
|
#define DMA_BASE_DST_ADR (0x8)
|
||
|
#define DMA_BYTE_CNT (0xC)
|
||
|
#define DMA_CURRENT_SRC_ADR (0x10)
|
||
|
#define DMA_CURRENT_DST_ADR (0x14)
|
||
|
#define DMA_CURRENT_BYTE_CNT (0x18)
|
||
|
#define DMA_INTR_ID (0x1C)
|
||
|
#define DMA_INTR_CLEAR_REG (DMA_CURRENT_SRC_ADR)
|
||
|
|
||
|
#define DMA_CALC_REG_ADR(channel, register) ((volatile u32*)(DMA_BASE + ((channel) << 5) + (register)))
|
||
|
|
||
|
#define DMA_CTRL_STATUS_FAIR_SHARE_ARB (1 << 0)
|
||
|
#define DMA_CTRL_STATUS_IN_PROGRESS (1 << 1)
|
||
|
#define DMA_CTRL_STATUS_SRC_DREQ_MASK (0x0000003C)
|
||
|
#define DMA_CTRL_STATUS_SRC_DREQ_SHIFT (2)
|
||
|
#define DMA_CTRL_STATUS_DEST_DREQ_MASK (0x000003C0)
|
||
|
#define DMA_CTRL_STATUS_DEST_DREQ_SHIFT (6)
|
||
|
#define DMA_CTRL_STATUS_INTR (1 << 10)
|
||
|
#define DMA_CTRL_STATUS_NXT_FREE (1 << 11)
|
||
|
#define DMA_CTRL_STATUS_RESET (1 << 12)
|
||
|
#define DMA_CTRL_STATUS_DIR_MASK (0x00006000)
|
||
|
#define DMA_CTRL_STATUS_DIR_SHIFT (13)
|
||
|
#define DMA_CTRL_STATUS_SRC_ADR_MODE (1 << 15)
|
||
|
#define DMA_CTRL_STATUS_DEST_ADR_MODE (1 << 16)
|
||
|
#define DMA_CTRL_STATUS_TRANSFER_MODE_A (1 << 17)
|
||
|
#define DMA_CTRL_STATUS_TRANSFER_MODE_B (1 << 18)
|
||
|
#define DMA_CTRL_STATUS_SRC_WIDTH_MASK (0x00380000)
|
||
|
#define DMA_CTRL_STATUS_SRC_WIDTH_SHIFT (19)
|
||
|
#define DMA_CTRL_STATUS_DEST_WIDTH_MASK (0x01C00000)
|
||
|
#define DMA_CTRL_STATUS_DEST_WIDTH_SHIFT (22)
|
||
|
#define DMA_CTRL_STATUS_PAUSE (1 << 25)
|
||
|
#define DMA_CTRL_STATUS_INTERRUPT_ENABLE (1 << 26)
|
||
|
#define DMA_CTRL_STATUS_SOURCE_ADDRESS_FIXED (1 << 27)
|
||
|
#define DMA_CTRL_STATUS_DESTINATION_ADDRESS_FIXED (1 << 28)
|
||
|
#define DMA_CTRL_STATUS_STARVE_LOW_PRIORITY (1 << 29)
|
||
|
#define DMA_CTRL_STATUS_INTR_CLEAR_ENABLE (1 << 30)
|
||
|
|
||
|
#define DMA_BYTE_CNT_MASK ((1 << 21) - 1)
|
||
|
#define DMA_BYTE_CNT_WR_EOT_MASK (1 << 30)
|
||
|
#define DMA_BYTE_CNT_RD_EOT_MASK (1 << 31)
|
||
|
#define DMA_BYTE_CNT_BURST_MASK (1 << 28)
|
||
|
|
||
|
#define MAKE_FIELD(value, num_bits, bit_num) (((value) & ((1 << (num_bits)) - 1)) << (bit_num))
|
||
|
|
||
|
typedef enum oxnas_dma_mode {
|
||
|
OXNAS_DMA_MODE_FIXED, OXNAS_DMA_MODE_INC
|
||
|
} oxnas_dma_mode_t;
|
||
|
|
||
|
typedef enum oxnas_dma_direction {
|
||
|
OXNAS_DMA_TO_DEVICE, OXNAS_DMA_FROM_DEVICE
|
||
|
} oxnas_dma_direction_t;
|
||
|
|
||
|
/* The available buses to which the DMA controller is attached */
|
||
|
typedef enum oxnas_dma_transfer_bus {
|
||
|
OXNAS_DMA_SIDE_A, OXNAS_DMA_SIDE_B
|
||
|
} oxnas_dma_transfer_bus_t;
|
||
|
|
||
|
/* Direction of data flow between the DMA controller's pair of interfaces */
|
||
|
typedef enum oxnas_dma_transfer_direction {
|
||
|
OXNAS_DMA_A_TO_A, OXNAS_DMA_B_TO_A, OXNAS_DMA_A_TO_B, OXNAS_DMA_B_TO_B
|
||
|
} oxnas_dma_transfer_direction_t;
|
||
|
|
||
|
/* The available data widths */
|
||
|
typedef enum oxnas_dma_transfer_width {
|
||
|
OXNAS_DMA_TRANSFER_WIDTH_8BITS,
|
||
|
OXNAS_DMA_TRANSFER_WIDTH_16BITS,
|
||
|
OXNAS_DMA_TRANSFER_WIDTH_32BITS
|
||
|
} oxnas_dma_transfer_width_t;
|
||
|
|
||
|
/* The mode of the DMA transfer */
|
||
|
typedef enum oxnas_dma_transfer_mode {
|
||
|
OXNAS_DMA_TRANSFER_MODE_SINGLE, OXNAS_DMA_TRANSFER_MODE_BURST
|
||
|
} oxnas_dma_transfer_mode_t;
|
||
|
|
||
|
/* The available transfer targets */
|
||
|
typedef enum oxnas_dma_dreq {
|
||
|
OXNAS_DMA_DREQ_SATA = 0, OXNAS_DMA_DREQ_MEMORY = 15
|
||
|
} oxnas_dma_dreq_t;
|
||
|
|
||
|
typedef struct oxnas_dma_device_settings {
|
||
|
unsigned long address_;
|
||
|
unsigned fifo_size_; // Chained transfers must take account of FIFO offset at end of previous transfer
|
||
|
unsigned char dreq_;
|
||
|
unsigned read_eot_ :1;
|
||
|
unsigned read_final_eot_ :1;
|
||
|
unsigned write_eot_ :1;
|
||
|
unsigned write_final_eot_ :1;
|
||
|
unsigned bus_ :1;
|
||
|
unsigned width_ :2;
|
||
|
unsigned transfer_mode_ :1;
|
||
|
unsigned address_mode_ :1;
|
||
|
unsigned address_really_fixed_ :1;
|
||
|
} oxnas_dma_device_settings_t;
|
||
|
|
||
|
static const int MAX_NO_ERROR_LOOPS = 100000; /* 1 second in units of 10uS */
|
||
|
static const int MAX_DMA_XFER_LOOPS = 300000; /* 30 seconds in units of 100uS */
|
||
|
static const int MAX_DMA_ABORT_LOOPS = 10000; /* 0.1 second in units of 10uS */
|
||
|
static const int MAX_SRC_READ_LOOPS = 10000; /* 0.1 second in units of 10uS */
|
||
|
static const int MAX_SRC_WRITE_LOOPS = 10000; /* 0.1 second in units of 10uS */
|
||
|
static const int MAX_NOT_BUSY_LOOPS = 10000; /* 1 second in units of 100uS */
|
||
|
|
||
|
/* The internal SATA drive on which we should attempt to find partitions */
|
||
|
static volatile u32* sata_regs_base[2] = { (volatile u32*) SATA_0_REGS_BASE,
|
||
|
(volatile u32*) SATA_1_REGS_BASE,
|
||
|
|
||
|
};
|
||
|
static u32 wr_sata_orb1[2] = { 0, 0 };
|
||
|
static u32 wr_sata_orb2[2] = { 0, 0 };
|
||
|
static u32 wr_sata_orb3[2] = { 0, 0 };
|
||
|
static u32 wr_sata_orb4[2] = { 0, 0 };
|
||
|
|
||
|
#ifdef CONFIG_LBA48
|
||
|
/* need keeping a record of NSECT LBAL LBAM LBAH ide_outb values for lba48 support */
|
||
|
#define OUT_HISTORY_BASE ATA_PORT_NSECT
|
||
|
#define OUT_HISTORY_MAX ATA_PORT_LBAH
|
||
|
static unsigned char out_history[2][OUT_HISTORY_MAX - OUT_HISTORY_BASE + 1] = {};
|
||
|
#endif
|
||
|
|
||
|
static oxnas_dma_device_settings_t oxnas_sata_dma_settings = { .address_ =
|
||
|
SATA_DATA_BASE, .fifo_size_ = 16, .dreq_ = OXNAS_DMA_DREQ_SATA,
|
||
|
.read_eot_ = 0, .read_final_eot_ = 1, .write_eot_ = 0,
|
||
|
.write_final_eot_ = 1, .bus_ = OXNAS_DMA_SIDE_B, .width_ =
|
||
|
OXNAS_DMA_TRANSFER_WIDTH_32BITS, .transfer_mode_ =
|
||
|
OXNAS_DMA_TRANSFER_MODE_BURST, .address_mode_ =
|
||
|
OXNAS_DMA_MODE_FIXED, .address_really_fixed_ = 0 };
|
||
|
|
||
|
oxnas_dma_device_settings_t oxnas_ram_dma_settings = { .address_ = 0,
|
||
|
.fifo_size_ = 0, .dreq_ = OXNAS_DMA_DREQ_MEMORY, .read_eot_ = 1,
|
||
|
.read_final_eot_ = 1, .write_eot_ = 1, .write_final_eot_ = 1,
|
||
|
.bus_ = OXNAS_DMA_SIDE_A, .width_ =
|
||
|
OXNAS_DMA_TRANSFER_WIDTH_32BITS, .transfer_mode_ =
|
||
|
OXNAS_DMA_TRANSFER_MODE_BURST, .address_mode_ =
|
||
|
OXNAS_DMA_MODE_FIXED, .address_really_fixed_ = 1 };
|
||
|
|
||
|
static void xfer_wr_shadow_to_orbs(int device)
|
||
|
{
|
||
|
*(sata_regs_base[device] + SATA_ORB1_OFF) = wr_sata_orb1[device];
|
||
|
*(sata_regs_base[device] + SATA_ORB2_OFF) = wr_sata_orb2[device];
|
||
|
*(sata_regs_base[device] + SATA_ORB3_OFF) = wr_sata_orb3[device];
|
||
|
*(sata_regs_base[device] + SATA_ORB4_OFF) = wr_sata_orb4[device];
|
||
|
}
|
||
|
|
||
|
static inline void device_select(int device)
|
||
|
{
|
||
|
/* master/slave has no meaning to SATA core */
|
||
|
}
|
||
|
|
||
|
static int disk_present[CONFIG_SYS_IDE_MAXDEVICE];
|
||
|
|
||
|
#include <ata.h>
|
||
|
|
||
|
unsigned char ide_inb(int device, int port)
|
||
|
{
|
||
|
unsigned char val = 0;
|
||
|
|
||
|
/* Only permit accesses to disks found to be present during ide_preinit() */
|
||
|
if (!disk_present[device]) {
|
||
|
return ATA_STAT_FAULT;
|
||
|
}
|
||
|
|
||
|
device_select(device);
|
||
|
|
||
|
switch (port) {
|
||
|
case ATA_PORT_CTL:
|
||
|
val = (*(sata_regs_base[device] + SATA_ORB4_OFF)
|
||
|
& (0xFFUL << SATA_CTL_BIT)) >> SATA_CTL_BIT;
|
||
|
break;
|
||
|
case ATA_PORT_FEATURE:
|
||
|
val = (*(sata_regs_base[device] + SATA_ORB2_OFF)
|
||
|
& (0xFFUL << SATA_FEATURE_BIT)) >> SATA_FEATURE_BIT;
|
||
|
break;
|
||
|
case ATA_PORT_NSECT:
|
||
|
val = (*(sata_regs_base[device] + SATA_ORB2_OFF)
|
||
|
& (0xFFUL << SATA_NSECT_BIT)) >> SATA_NSECT_BIT;
|
||
|
break;
|
||
|
case ATA_PORT_LBAL:
|
||
|
val = (*(sata_regs_base[device] + SATA_ORB3_OFF)
|
||
|
& (0xFFUL << SATA_LBAL_BIT)) >> SATA_LBAL_BIT;
|
||
|
break;
|
||
|
case ATA_PORT_LBAM:
|
||
|
val = (*(sata_regs_base[device] + SATA_ORB3_OFF)
|
||
|
& (0xFFUL << SATA_LBAM_BIT)) >> SATA_LBAM_BIT;
|
||
|
break;
|
||
|
case ATA_PORT_LBAH:
|
||
|
val = (*(sata_regs_base[device] + SATA_ORB3_OFF)
|
||
|
& (0xFFUL << SATA_LBAH_BIT)) >> SATA_LBAH_BIT;
|
||
|
break;
|
||
|
case ATA_PORT_DEVICE:
|
||
|
val = (*(sata_regs_base[device] + SATA_ORB3_OFF)
|
||
|
& (0xFFUL << SATA_HOB_LBAH_BIT)) >> SATA_HOB_LBAH_BIT;
|
||
|
val |= (*(sata_regs_base[device] + SATA_ORB1_OFF)
|
||
|
& (0xFFUL << SATA_DEVICE_BIT)) >> SATA_DEVICE_BIT;
|
||
|
break;
|
||
|
case ATA_PORT_COMMAND:
|
||
|
val = (*(sata_regs_base[device] + SATA_ORB2_OFF)
|
||
|
& (0xFFUL << SATA_COMMAND_BIT)) >> SATA_COMMAND_BIT;
|
||
|
val |= ATA_STAT_DRQ;
|
||
|
break;
|
||
|
default:
|
||
|
printf("ide_inb() Unknown port = %d\n", port);
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
// printf("inb: %d:%01x => %02x\n", device, port, val);
|
||
|
|
||
|
return val;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Possible that ATA status will not become no-error, so must have timeout
|
||
|
* @returns An int which is zero on error
|
||
|
*/
|
||
|
static inline int wait_no_error(int device)
|
||
|
{
|
||
|
int status = 0;
|
||
|
|
||
|
/* Check for ATA core error */
|
||
|
if (*(sata_regs_base[device] + SATA_INT_STATUS_OFF)
|
||
|
& (1 << SATA_INT_STATUS_ERROR_BIT)) {
|
||
|
printf("wait_no_error() SATA core flagged error\n");
|
||
|
} else {
|
||
|
int loops = MAX_NO_ERROR_LOOPS;
|
||
|
do {
|
||
|
/* Check for ATA device error */
|
||
|
if (!(ide_inb(device, ATA_PORT_COMMAND)
|
||
|
& (1 << ATA_STATUS_ERR_BIT))) {
|
||
|
status = 1;
|
||
|
break;
|
||
|
}
|
||
|
udelay(10);
|
||
|
} while (--loops);
|
||
|
|
||
|
if (!loops) {
|
||
|
printf("wait_no_error() Timed out of wait for SATA no-error condition\n");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return status;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Expect SATA command to always finish, perhaps with error
|
||
|
* @returns An int which is zero on error
|
||
|
*/
|
||
|
static inline int wait_sata_command_not_busy(int device)
|
||
|
{
|
||
|
/* Wait for data to be available */
|
||
|
int status = 0;
|
||
|
int loops = MAX_NOT_BUSY_LOOPS;
|
||
|
do {
|
||
|
if (!(*(sata_regs_base[device] + SATA_COMMAND_OFF)
|
||
|
& (1 << SATA_CMD_BUSY_BIT))) {
|
||
|
status = 1;
|
||
|
break;
|
||
|
}
|
||
|
udelay(100);
|
||
|
} while (--loops);
|
||
|
|
||
|
if (!loops) {
|
||
|
printf("wait_sata_command_not_busy() Timed out of wait for SATA command to finish\n");
|
||
|
}
|
||
|
|
||
|
return status;
|
||
|
}
|
||
|
|
||
|
void ide_outb(int device, int port, unsigned char val)
|
||
|
{
|
||
|
typedef enum send_method {
|
||
|
SEND_NONE, SEND_SIMPLE, SEND_CMD, SEND_CTL,
|
||
|
} send_method_t;
|
||
|
|
||
|
/* Only permit accesses to disks found to be present during ide_preinit() */
|
||
|
if (!disk_present[device]) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
// printf("outb: %d:%01x <= %02x\n", device, port, val);
|
||
|
|
||
|
device_select(device);
|
||
|
|
||
|
#ifdef CONFIG_LBA48
|
||
|
if (port >= OUT_HISTORY_BASE && port <= OUT_HISTORY_MAX) {
|
||
|
out_history[0][port - OUT_HISTORY_BASE] =
|
||
|
out_history[1][port - OUT_HISTORY_BASE];
|
||
|
out_history[1][port - OUT_HISTORY_BASE] = val;
|
||
|
}
|
||
|
#endif
|
||
|
send_method_t send_regs = SEND_NONE;
|
||
|
switch (port) {
|
||
|
case ATA_PORT_CTL:
|
||
|
wr_sata_orb4[device] &= ~(0xFFUL << SATA_CTL_BIT);
|
||
|
wr_sata_orb4[device] |= (val << SATA_CTL_BIT);
|
||
|
send_regs = SEND_CTL;
|
||
|
break;
|
||
|
case ATA_PORT_FEATURE:
|
||
|
wr_sata_orb2[device] &= ~(0xFFUL << SATA_FEATURE_BIT);
|
||
|
wr_sata_orb2[device] |= (val << SATA_FEATURE_BIT);
|
||
|
send_regs = SEND_SIMPLE;
|
||
|
break;
|
||
|
case ATA_PORT_NSECT:
|
||
|
wr_sata_orb2[device] &= ~(0xFFUL << SATA_NSECT_BIT);
|
||
|
wr_sata_orb2[device] |= (val << SATA_NSECT_BIT);
|
||
|
send_regs = SEND_SIMPLE;
|
||
|
break;
|
||
|
case ATA_PORT_LBAL:
|
||
|
wr_sata_orb3[device] &= ~(0xFFUL << SATA_LBAL_BIT);
|
||
|
wr_sata_orb3[device] |= (val << SATA_LBAL_BIT);
|
||
|
send_regs = SEND_SIMPLE;
|
||
|
break;
|
||
|
case ATA_PORT_LBAM:
|
||
|
wr_sata_orb3[device] &= ~(0xFFUL << SATA_LBAM_BIT);
|
||
|
wr_sata_orb3[device] |= (val << SATA_LBAM_BIT);
|
||
|
send_regs = SEND_SIMPLE;
|
||
|
break;
|
||
|
case ATA_PORT_LBAH:
|
||
|
wr_sata_orb3[device] &= ~(0xFFUL << SATA_LBAH_BIT);
|
||
|
wr_sata_orb3[device] |= (val << SATA_LBAH_BIT);
|
||
|
send_regs = SEND_SIMPLE;
|
||
|
break;
|
||
|
case ATA_PORT_DEVICE:
|
||
|
wr_sata_orb1[device] &= ~(0xFFUL << SATA_DEVICE_BIT);
|
||
|
wr_sata_orb1[device] |= (val << SATA_DEVICE_BIT);
|
||
|
send_regs = SEND_SIMPLE;
|
||
|
break;
|
||
|
case ATA_PORT_COMMAND:
|
||
|
wr_sata_orb2[device] &= ~(0xFFUL << SATA_COMMAND_BIT);
|
||
|
wr_sata_orb2[device] |= (val << SATA_COMMAND_BIT);
|
||
|
send_regs = SEND_CMD;
|
||
|
#ifdef CONFIG_LBA48
|
||
|
if (val == ATA_CMD_READ_EXT || val == ATA_CMD_WRITE_EXT)
|
||
|
{
|
||
|
/* fill high bytes of LBA48 && NSECT */
|
||
|
wr_sata_orb2[device] &= ~(0xFFUL << SATA_HOB_NSECT_BIT);
|
||
|
wr_sata_orb2[device] |=
|
||
|
(out_history[0][ATA_PORT_NSECT - OUT_HISTORY_BASE] << SATA_HOB_NSECT_BIT);
|
||
|
|
||
|
wr_sata_orb3[device] &= ~(0xFFUL << SATA_HOB_LBAH_BIT);
|
||
|
wr_sata_orb3[device] |=
|
||
|
(out_history[0][ATA_PORT_LBAL - OUT_HISTORY_BASE] << SATA_HOB_LBAH_BIT);
|
||
|
|
||
|
wr_sata_orb4[device] &= ~(0xFFUL << SATA_LBA32_BIT);
|
||
|
wr_sata_orb4[device] |=
|
||
|
(out_history[0][ATA_PORT_LBAM - OUT_HISTORY_BASE] << SATA_LBA32_BIT);
|
||
|
|
||
|
wr_sata_orb4[device] &= ~(0xFFUL << SATA_LBA40_BIT);
|
||
|
wr_sata_orb4[device] |=
|
||
|
(out_history[0][ATA_PORT_LBAH - OUT_HISTORY_BASE] << SATA_LBA40_BIT);
|
||
|
}
|
||
|
#endif
|
||
|
break;
|
||
|
default:
|
||
|
printf("ide_outb() Unknown port = %d\n", port);
|
||
|
}
|
||
|
|
||
|
u32 command;
|
||
|
switch (send_regs) {
|
||
|
case SEND_CMD:
|
||
|
wait_sata_command_not_busy(device);
|
||
|
command = *(sata_regs_base[device] + SATA_COMMAND_OFF);
|
||
|
command &= ~SATA_OPCODE_MASK;
|
||
|
command |= SATA_CMD_WRITE_TO_ORB_REGS;
|
||
|
xfer_wr_shadow_to_orbs(device);
|
||
|
wait_sata_command_not_busy(device);
|
||
|
*(sata_regs_base[device] + SATA_COMMAND_OFF) = command;
|
||
|
if (!wait_no_error(device)) {
|
||
|
printf("ide_outb() Wait for ATA no-error timed-out\n");
|
||
|
}
|
||
|
break;
|
||
|
case SEND_CTL:
|
||
|
wait_sata_command_not_busy(device);
|
||
|
command = *(sata_regs_base[device] + SATA_COMMAND_OFF);
|
||
|
command &= ~SATA_OPCODE_MASK;
|
||
|
command |= SATA_CMD_WRITE_TO_ORB_REGS_NO_COMMAND;
|
||
|
xfer_wr_shadow_to_orbs(device);
|
||
|
wait_sata_command_not_busy(device);
|
||
|
*(sata_regs_base[device] + SATA_COMMAND_OFF) = command;
|
||
|
if (!wait_no_error(device)) {
|
||
|
printf("ide_outb() Wait for ATA no-error timed-out\n");
|
||
|
}
|
||
|
break;
|
||
|
default:
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static u32 encode_start(u32 ctrl_status)
|
||
|
{
|
||
|
return ctrl_status & ~DMA_CTRL_STATUS_PAUSE;
|
||
|
}
|
||
|
|
||
|
/* start a paused DMA transfer in channel 0 of the SATA DMA core */
|
||
|
static void dma_start(void)
|
||
|
{
|
||
|
unsigned int reg;
|
||
|
reg = readl(SATA_DMA_REGS_BASE + DMA_CTRL_STATUS);
|
||
|
reg = encode_start(reg);
|
||
|
writel(reg, SATA_DMA_REGS_BASE + DMA_CTRL_STATUS);
|
||
|
}
|
||
|
|
||
|
static unsigned long encode_control_status(
|
||
|
oxnas_dma_device_settings_t* src_settings,
|
||
|
oxnas_dma_device_settings_t* dst_settings)
|
||
|
{
|
||
|
unsigned long ctrl_status;
|
||
|
oxnas_dma_transfer_direction_t direction;
|
||
|
|
||
|
ctrl_status = DMA_CTRL_STATUS_PAUSE; // Paused
|
||
|
ctrl_status |= DMA_CTRL_STATUS_FAIR_SHARE_ARB; // High priority
|
||
|
ctrl_status |= (src_settings->dreq_ << DMA_CTRL_STATUS_SRC_DREQ_SHIFT); // Dreq
|
||
|
ctrl_status |= (dst_settings->dreq_ << DMA_CTRL_STATUS_DEST_DREQ_SHIFT); // Dreq
|
||
|
ctrl_status &= ~DMA_CTRL_STATUS_RESET; // !RESET
|
||
|
|
||
|
// Use new interrupt clearing register
|
||
|
ctrl_status |= DMA_CTRL_STATUS_INTR_CLEAR_ENABLE;
|
||
|
|
||
|
// Setup the transfer direction and burst/single mode for the two DMA busses
|
||
|
if (src_settings->bus_ == OXNAS_DMA_SIDE_A) {
|
||
|
// Set the burst/single mode for bus A based on src device's settings
|
||
|
if (src_settings->transfer_mode_
|
||
|
== OXNAS_DMA_TRANSFER_MODE_BURST) {
|
||
|
ctrl_status |= DMA_CTRL_STATUS_TRANSFER_MODE_A;
|
||
|
} else {
|
||
|
ctrl_status &= ~DMA_CTRL_STATUS_TRANSFER_MODE_A;
|
||
|
}
|
||
|
|
||
|
if (dst_settings->bus_ == OXNAS_DMA_SIDE_A) {
|
||
|
direction = OXNAS_DMA_A_TO_A;
|
||
|
} else {
|
||
|
direction = OXNAS_DMA_A_TO_B;
|
||
|
|
||
|
// Set the burst/single mode for bus B based on dst device's settings
|
||
|
if (dst_settings->transfer_mode_
|
||
|
== OXNAS_DMA_TRANSFER_MODE_BURST) {
|
||
|
ctrl_status |= DMA_CTRL_STATUS_TRANSFER_MODE_B;
|
||
|
} else {
|
||
|
ctrl_status &= ~DMA_CTRL_STATUS_TRANSFER_MODE_B;
|
||
|
}
|
||
|
}
|
||
|
} else {
|
||
|
// Set the burst/single mode for bus B based on src device's settings
|
||
|
if (src_settings->transfer_mode_
|
||
|
== OXNAS_DMA_TRANSFER_MODE_BURST) {
|
||
|
ctrl_status |= DMA_CTRL_STATUS_TRANSFER_MODE_B;
|
||
|
} else {
|
||
|
ctrl_status &= ~DMA_CTRL_STATUS_TRANSFER_MODE_B;
|
||
|
}
|
||
|
|
||
|
if (dst_settings->bus_ == OXNAS_DMA_SIDE_A) {
|
||
|
direction = OXNAS_DMA_B_TO_A;
|
||
|
|
||
|
// Set the burst/single mode for bus A based on dst device's settings
|
||
|
if (dst_settings->transfer_mode_
|
||
|
== OXNAS_DMA_TRANSFER_MODE_BURST) {
|
||
|
ctrl_status |= DMA_CTRL_STATUS_TRANSFER_MODE_A;
|
||
|
} else {
|
||
|
ctrl_status &= ~DMA_CTRL_STATUS_TRANSFER_MODE_A;
|
||
|
}
|
||
|
} else {
|
||
|
direction = OXNAS_DMA_B_TO_B;
|
||
|
}
|
||
|
}
|
||
|
ctrl_status |= (direction << DMA_CTRL_STATUS_DIR_SHIFT);
|
||
|
|
||
|
// Setup source address mode fixed or increment
|
||
|
if (src_settings->address_mode_ == OXNAS_DMA_MODE_FIXED) {
|
||
|
// Fixed address
|
||
|
ctrl_status &= ~(DMA_CTRL_STATUS_SRC_ADR_MODE);
|
||
|
|
||
|
// Set up whether fixed address is _really_ fixed
|
||
|
if (src_settings->address_really_fixed_) {
|
||
|
ctrl_status |= DMA_CTRL_STATUS_SOURCE_ADDRESS_FIXED;
|
||
|
} else {
|
||
|
ctrl_status &= ~DMA_CTRL_STATUS_SOURCE_ADDRESS_FIXED;
|
||
|
}
|
||
|
} else {
|
||
|
// Incrementing address
|
||
|
ctrl_status |= DMA_CTRL_STATUS_SRC_ADR_MODE;
|
||
|
ctrl_status &= ~DMA_CTRL_STATUS_SOURCE_ADDRESS_FIXED;
|
||
|
}
|
||
|
|
||
|
// Setup destination address mode fixed or increment
|
||
|
if (dst_settings->address_mode_ == OXNAS_DMA_MODE_FIXED) {
|
||
|
// Fixed address
|
||
|
ctrl_status &= ~(DMA_CTRL_STATUS_DEST_ADR_MODE);
|
||
|
|
||
|
// Set up whether fixed address is _really_ fixed
|
||
|
if (dst_settings->address_really_fixed_) {
|
||
|
ctrl_status |=
|
||
|
DMA_CTRL_STATUS_DESTINATION_ADDRESS_FIXED;
|
||
|
} else {
|
||
|
ctrl_status &=
|
||
|
~DMA_CTRL_STATUS_DESTINATION_ADDRESS_FIXED;
|
||
|
}
|
||
|
} else {
|
||
|
// Incrementing address
|
||
|
ctrl_status |= DMA_CTRL_STATUS_DEST_ADR_MODE;
|
||
|
ctrl_status &= ~DMA_CTRL_STATUS_DESTINATION_ADDRESS_FIXED;
|
||
|
}
|
||
|
|
||
|
// Set up the width of the transfers on the DMA buses
|
||
|
ctrl_status |=
|
||
|
(src_settings->width_ << DMA_CTRL_STATUS_SRC_WIDTH_SHIFT);
|
||
|
ctrl_status |=
|
||
|
(dst_settings->width_ << DMA_CTRL_STATUS_DEST_WIDTH_SHIFT);
|
||
|
|
||
|
// Setup the priority arbitration scheme
|
||
|
ctrl_status &= ~DMA_CTRL_STATUS_STARVE_LOW_PRIORITY; // !Starve low priority
|
||
|
|
||
|
return ctrl_status;
|
||
|
}
|
||
|
|
||
|
static u32 encode_final_eot(oxnas_dma_device_settings_t* src_settings,
|
||
|
oxnas_dma_device_settings_t* dst_settings,
|
||
|
unsigned long length)
|
||
|
{
|
||
|
// Write the length, with EOT configuration for a final transfer
|
||
|
unsigned long encoded = length;
|
||
|
if (dst_settings->write_final_eot_) {
|
||
|
encoded |= DMA_BYTE_CNT_WR_EOT_MASK;
|
||
|
} else {
|
||
|
encoded &= ~DMA_BYTE_CNT_WR_EOT_MASK;
|
||
|
}
|
||
|
if (src_settings->read_final_eot_) {
|
||
|
encoded |= DMA_BYTE_CNT_RD_EOT_MASK;
|
||
|
} else {
|
||
|
encoded &= ~DMA_BYTE_CNT_RD_EOT_MASK;
|
||
|
}
|
||
|
/* if((src_settings->transfer_mode_) ||
|
||
|
(src_settings->transfer_mode_)) {
|
||
|
encoded |= DMA_BYTE_CNT_BURST_MASK;
|
||
|
} else {
|
||
|
encoded &= ~DMA_BYTE_CNT_BURST_MASK;
|
||
|
}*/
|
||
|
return encoded;
|
||
|
}
|
||
|
|
||
|
static void dma_start_write(const ulong* buffer, int num_bytes)
|
||
|
{
|
||
|
// Assemble complete memory settings
|
||
|
oxnas_dma_device_settings_t mem_settings = oxnas_ram_dma_settings;
|
||
|
mem_settings.address_ = (unsigned long) buffer;
|
||
|
mem_settings.address_mode_ = OXNAS_DMA_MODE_INC;
|
||
|
|
||
|
writel(encode_control_status(&mem_settings, &oxnas_sata_dma_settings),
|
||
|
SATA_DMA_REGS_BASE + DMA_CTRL_STATUS);
|
||
|
writel(mem_settings.address_, SATA_DMA_REGS_BASE + DMA_BASE_SRC_ADR);
|
||
|
writel(oxnas_sata_dma_settings.address_,
|
||
|
SATA_DMA_REGS_BASE + DMA_BASE_DST_ADR);
|
||
|
writel(encode_final_eot(&mem_settings, &oxnas_sata_dma_settings,
|
||
|
num_bytes),
|
||
|
SATA_DMA_REGS_BASE + DMA_BYTE_CNT);
|
||
|
|
||
|
dma_start();
|
||
|
}
|
||
|
|
||
|
static void dma_start_read(ulong* buffer, int num_bytes)
|
||
|
{
|
||
|
// Assemble complete memory settings
|
||
|
oxnas_dma_device_settings_t mem_settings = oxnas_ram_dma_settings;
|
||
|
mem_settings.address_ = (unsigned long) buffer;
|
||
|
mem_settings.address_mode_ = OXNAS_DMA_MODE_INC;
|
||
|
|
||
|
writel(encode_control_status(&oxnas_sata_dma_settings, &mem_settings),
|
||
|
SATA_DMA_REGS_BASE + DMA_CTRL_STATUS);
|
||
|
writel(oxnas_sata_dma_settings.address_,
|
||
|
SATA_DMA_REGS_BASE + DMA_BASE_SRC_ADR);
|
||
|
writel(mem_settings.address_, SATA_DMA_REGS_BASE + DMA_BASE_DST_ADR);
|
||
|
writel(encode_final_eot(&oxnas_sata_dma_settings, &mem_settings,
|
||
|
num_bytes),
|
||
|
SATA_DMA_REGS_BASE + DMA_BYTE_CNT);
|
||
|
|
||
|
dma_start();
|
||
|
}
|
||
|
|
||
|
static inline int dma_busy(void)
|
||
|
{
|
||
|
return readl(SATA_DMA_REGS_BASE + DMA_CTRL_STATUS)
|
||
|
& DMA_CTRL_STATUS_IN_PROGRESS;
|
||
|
}
|
||
|
|
||
|
static int wait_dma_not_busy(int device)
|
||
|
{
|
||
|
unsigned int cleanup_required = 0;
|
||
|
|
||
|
/* Poll for DMA completion */
|
||
|
int loops = MAX_DMA_XFER_LOOPS;
|
||
|
do {
|
||
|
if (!dma_busy()) {
|
||
|
break;
|
||
|
}
|
||
|
udelay(100);
|
||
|
} while (--loops);
|
||
|
|
||
|
if (!loops) {
|
||
|
printf("wait_dma_not_busy() Timed out of wait for DMA not busy\n");
|
||
|
cleanup_required = 1;
|
||
|
}
|
||
|
|
||
|
if (cleanup_required) {
|
||
|
/* Abort DMA to make sure it has finished. */
|
||
|
unsigned int ctrl_status = readl(
|
||
|
SATA_DMA_CHANNEL + DMA_CTRL_STATUS);
|
||
|
ctrl_status |= DMA_CTRL_STATUS_RESET;
|
||
|
writel(ctrl_status, SATA_DMA_CHANNEL + DMA_CTRL_STATUS);
|
||
|
|
||
|
// Wait for the channel to become idle - should be quick as should
|
||
|
// finish after the next AHB single or burst transfer
|
||
|
loops = MAX_DMA_ABORT_LOOPS;
|
||
|
do {
|
||
|
if (!dma_busy()) {
|
||
|
break;
|
||
|
}
|
||
|
udelay(10);
|
||
|
} while (--loops);
|
||
|
|
||
|
if (!loops) {
|
||
|
printf("wait_dma_not_busy() Timed out of wait for DMA channel abort\n");
|
||
|
} else {
|
||
|
/* Successfully cleanup the DMA channel */
|
||
|
cleanup_required = 0;
|
||
|
}
|
||
|
|
||
|
// Deassert reset for the channel
|
||
|
ctrl_status = readl(SATA_DMA_CHANNEL + DMA_CTRL_STATUS);
|
||
|
ctrl_status &= ~DMA_CTRL_STATUS_RESET;
|
||
|
writel(ctrl_status, SATA_DMA_CHANNEL + DMA_CTRL_STATUS);
|
||
|
}
|
||
|
|
||
|
return !cleanup_required;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Possible that ATA status will not become not-busy, so must have timeout
|
||
|
*/
|
||
|
static unsigned int wait_not_busy(int device, unsigned long timeout_secs)
|
||
|
{
|
||
|
int busy = 1;
|
||
|
unsigned long loops = (timeout_secs * 1000) / 50;
|
||
|
do {
|
||
|
// Test the ATA status register BUSY flag
|
||
|
if (!((*(sata_regs_base[device] + SATA_ORB2_OFF)
|
||
|
>> SATA_COMMAND_BIT) & (1UL << ATA_STATUS_BSY_BIT))) {
|
||
|
/* Not busy, so stop polling */
|
||
|
busy = 0;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
// Wait for 50mS before sampling ATA status register again
|
||
|
udelay(50000);
|
||
|
} while (--loops);
|
||
|
|
||
|
return busy;
|
||
|
}
|
||
|
|
||
|
void ide_output_data(int device, const ulong *sect_buf, int words)
|
||
|
{
|
||
|
/* Only permit accesses to disks found to be present during ide_preinit() */
|
||
|
if (!disk_present[device]) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
/* Select the required internal SATA drive */
|
||
|
device_select(device);
|
||
|
|
||
|
/* Start the DMA channel sending data from the passed buffer to the SATA core */
|
||
|
dma_start_write(sect_buf, words << 2);
|
||
|
|
||
|
/* Don't know why we need this delay, but without it the wait for DMA not
|
||
|
busy times soemtimes out, e.g. when saving environment to second disk */
|
||
|
udelay(1000);
|
||
|
|
||
|
/* Wait for DMA to finish */
|
||
|
if (!wait_dma_not_busy(device)) {
|
||
|
printf("Timed out of wait for DMA channel for SATA device %d to have in-progress clear\n",
|
||
|
device);
|
||
|
}
|
||
|
|
||
|
/* Sata core should finish after DMA */
|
||
|
if (wait_not_busy(device, 30)) {
|
||
|
printf("Timed out of wait for SATA device %d to have BUSY clear\n",
|
||
|
device);
|
||
|
}
|
||
|
if (!wait_no_error(device)) {
|
||
|
printf("oxnas_sata_output_data() Wait for ATA no-error timed-out\n");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
#define SATA_DM_DBG1 (SATA_HOST_REGS_BASE + 0)
|
||
|
#define SATA_DATACOUNT_PORT0 (SATA_HOST_REGS_BASE + 0x10)
|
||
|
#define SATA_DATACOUNT_PORT1 (SATA_HOST_REGS_BASE + 0x14)
|
||
|
#define SATA_DATA_MUX_RAM0 (SATA_HOST_REGS_BASE + 0x8000)
|
||
|
#define SATA_DATA_MUX_RAM1 (SATA_HOST_REGS_BASE + 0xA000)
|
||
|
/* Sata core debug1 register bits */
|
||
|
#define SATA_CORE_PORT0_DATA_DIR_BIT 20
|
||
|
#define SATA_CORE_PORT1_DATA_DIR_BIT 21
|
||
|
#define SATA_CORE_PORT0_DATA_DIR (1 << SATA_CORE_PORT0_DATA_DIR_BIT)
|
||
|
#define SATA_CORE_PORT1_DATA_DIR (1 << SATA_CORE_PORT1_DATA_DIR_BIT)
|
||
|
|
||
|
/**
|
||
|
* Ref bug-6320
|
||
|
*
|
||
|
* This code is a work around for a DMA hardware bug that will repeat the
|
||
|
* penultimate 8-bytes on some reads. This code will check that the amount
|
||
|
* of data transferred is a multiple of 512 bytes, if not the in it will
|
||
|
* fetch the correct data from a buffer in the SATA core and copy it into
|
||
|
* memory.
|
||
|
*
|
||
|
*/
|
||
|
static void sata_bug_6320_workaround(int port, ulong *candidate)
|
||
|
{
|
||
|
int is_read;
|
||
|
int quads_transferred;
|
||
|
int remainder;
|
||
|
int sector_quads_remaining;
|
||
|
|
||
|
/* Only want to apply fix to reads */
|
||
|
is_read = !(*((unsigned long*) SATA_DM_DBG1)
|
||
|
& (port ? SATA_CORE_PORT1_DATA_DIR : SATA_CORE_PORT0_DATA_DIR));
|
||
|
|
||
|
/* Check for an incomplete transfer, i.e. not a multiple of 512 bytes
|
||
|
transferred (datacount_port register counts quads transferred) */
|
||
|
quads_transferred = *((unsigned long*) (
|
||
|
port ? SATA_DATACOUNT_PORT1 : SATA_DATACOUNT_PORT0));
|
||
|
|
||
|
remainder = quads_transferred & 0x7f;
|
||
|
sector_quads_remaining = remainder ? (0x80 - remainder) : 0;
|
||
|
|
||
|
if (is_read && (sector_quads_remaining == 2)) {
|
||
|
debug("SATA read fixup, only transfered %d quads, "
|
||
|
"sector_quads_remaining %d, port %d\n",
|
||
|
quads_transferred, sector_quads_remaining, port);
|
||
|
|
||
|
int total_len = ATA_SECT_SIZE;
|
||
|
ulong *sata_data_ptr = (void*) (
|
||
|
port ? SATA_DATA_MUX_RAM1 : SATA_DATA_MUX_RAM0)
|
||
|
+ ((total_len - 8) % 2048);
|
||
|
|
||
|
*candidate = *sata_data_ptr;
|
||
|
*(candidate + 1) = *(sata_data_ptr + 1);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
void ide_input_data(int device, ulong *sect_buf, int words)
|
||
|
{
|
||
|
/* Only permit accesses to disks found to be present during ide_preinit() */
|
||
|
if (!disk_present[device]) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
/* Select the required internal SATA drive */
|
||
|
device_select(device);
|
||
|
|
||
|
/* Start the DMA channel receiving data from the SATA core into the passed buffer */
|
||
|
dma_start_read(sect_buf, words << 2);
|
||
|
|
||
|
/* Sata core should finish before DMA */
|
||
|
if (wait_not_busy(device, 30)) {
|
||
|
printf("Timed out of wait for SATA device %d to have BUSY clear\n",
|
||
|
device);
|
||
|
}
|
||
|
if (!wait_no_error(device)) {
|
||
|
printf("oxnas_sata_output_data() Wait for ATA no-error timed-out\n");
|
||
|
}
|
||
|
|
||
|
/* Wait for DMA to finish */
|
||
|
if (!wait_dma_not_busy(device)) {
|
||
|
printf("Timed out of wait for DMA channel for SATA device %d to have in-progress clear\n",
|
||
|
device);
|
||
|
}
|
||
|
|
||
|
if (words == ATA_SECTORWORDS)
|
||
|
sata_bug_6320_workaround(device, sect_buf + words - 2);
|
||
|
}
|
||
|
|
||
|
static u32 scr_read(int device, unsigned int sc_reg)
|
||
|
{
|
||
|
/* Setup adr of required register. std regs start eight into async region */
|
||
|
*(sata_regs_base[device] + SATA_LINK_RD_ADDR) = sc_reg
|
||
|
* 4+ SATA_STD_ASYNC_REGS_OFF;
|
||
|
|
||
|
/* Wait for data to be available */
|
||
|
int loops = MAX_SRC_READ_LOOPS;
|
||
|
do {
|
||
|
if (*(sata_regs_base[device] + SATA_LINK_CONTROL) & 1UL) {
|
||
|
break;
|
||
|
}
|
||
|
udelay(10);
|
||
|
} while (--loops);
|
||
|
|
||
|
if (!loops) {
|
||
|
printf("scr_read() Timed out of wait for read completion\n");
|
||
|
}
|
||
|
|
||
|
/* Read the data from the async register */
|
||
|
return *(sata_regs_base[device] + SATA_LINK_DATA);
|
||
|
}
|
||
|
|
||
|
static void scr_write(int device, unsigned int sc_reg, u32 val)
|
||
|
{
|
||
|
/* Setup the data for the write */
|
||
|
*(sata_regs_base[device] + SATA_LINK_DATA) = val;
|
||
|
|
||
|
/* Setup adr of required register. std regs start eight into async region */
|
||
|
*(sata_regs_base[device] + SATA_LINK_WR_ADDR) = sc_reg
|
||
|
* 4+ SATA_STD_ASYNC_REGS_OFF;
|
||
|
|
||
|
/* Wait for data to be written */
|
||
|
int loops = MAX_SRC_WRITE_LOOPS;
|
||
|
do {
|
||
|
if (*(sata_regs_base[device] + SATA_LINK_CONTROL) & 1UL) {
|
||
|
break;
|
||
|
}
|
||
|
udelay(10);
|
||
|
} while (--loops);
|
||
|
|
||
|
if (!loops) {
|
||
|
printf("scr_write() Timed out of wait for write completion\n");
|
||
|
}
|
||
|
}
|
||
|
extern void workaround5458(void);
|
||
|
|
||
|
#define PHY_LOOP_COUNT 25 /* Wait for upto 5 seconds for PHY to be found */
|
||
|
#define LOS_AND_TX_LVL 0x2988
|
||
|
#define TX_ATTEN 0x55629
|
||
|
|
||
|
static int phy_reset(int device)
|
||
|
{
|
||
|
int phy_status = 0;
|
||
|
int loops = 0;
|
||
|
|
||
|
scr_write(device, (0x60 - SATA_STD_ASYNC_REGS_OFF) / 4, LOS_AND_TX_LVL);
|
||
|
scr_write(device, (0x70 - SATA_STD_ASYNC_REGS_OFF) / 4, TX_ATTEN);
|
||
|
|
||
|
/* limit it to Gen-1 SATA (1.5G) */
|
||
|
scr_write(device, SATA_SCR_CONTROL, 0x311); /* Issue phy wake & core reset */
|
||
|
scr_read(device, SATA_SCR_STATUS); /* Dummy read; flush */
|
||
|
udelay(1000);
|
||
|
scr_write(device, SATA_SCR_CONTROL, 0x310); /* Issue phy wake & clear core reset */
|
||
|
|
||
|
/* Wait for upto 5 seconds for PHY to become ready */
|
||
|
do {
|
||
|
udelay(200000);
|
||
|
if ((scr_read(device, SATA_SCR_STATUS) & 0xf) == 3) {
|
||
|
scr_write(device, SATA_SCR_ERROR, ~0);
|
||
|
phy_status = 1;
|
||
|
break;
|
||
|
}
|
||
|
//printf("No SATA PHY found status:0x%x\n", scr_read(device, SATA_SCR_STATUS));
|
||
|
} while (++loops < PHY_LOOP_COUNT);
|
||
|
|
||
|
if (phy_status) {
|
||
|
udelay(500000); /* wait half a second */
|
||
|
}
|
||
|
|
||
|
return phy_status;
|
||
|
}
|
||
|
|
||
|
#define FIS_LOOP_COUNT 25 /* Wait for upto 5 seconds for FIS to be received */
|
||
|
static int wait_FIS(int device)
|
||
|
{
|
||
|
int status = 0;
|
||
|
int loops = 0;
|
||
|
|
||
|
do {
|
||
|
udelay(200000);
|
||
|
if (ide_inb(device, ATA_PORT_NSECT) > 0) {
|
||
|
status = 1;
|
||
|
break;
|
||
|
}
|
||
|
} while (++loops < FIS_LOOP_COUNT);
|
||
|
|
||
|
return status;
|
||
|
}
|
||
|
|
||
|
|
||
|
#define SATA_PHY_ASIC_STAT (0x44900000)
|
||
|
#define SATA_PHY_ASIC_DATA (0x44900004)
|
||
|
|
||
|
/**
|
||
|
* initialise functions and macros for ASIC implementation
|
||
|
*/
|
||
|
#define PH_GAIN 2
|
||
|
#define FR_GAIN 3
|
||
|
#define PH_GAIN_OFFSET 6
|
||
|
#define FR_GAIN_OFFSET 8
|
||
|
#define PH_GAIN_MASK (0x3 << PH_GAIN_OFFSET)
|
||
|
#define FR_GAIN_MASK (0x3 << FR_GAIN_OFFSET)
|
||
|
#define USE_INT_SETTING (1<<5)
|
||
|
|
||
|
#define CR_READ_ENABLE (1<<16)
|
||
|
#define CR_WRITE_ENABLE (1<<17)
|
||
|
#define CR_CAP_DATA (1<<18)
|
||
|
|
||
|
static void wait_cr_ack(void)
|
||
|
{
|
||
|
while ((readl(SATA_PHY_ASIC_STAT) >> 16) & 0x1f)
|
||
|
/* wait for an ack bit to be set */;
|
||
|
}
|
||
|
|
||
|
static unsigned short read_cr(unsigned short address)
|
||
|
{
|
||
|
writel(address, SATA_PHY_ASIC_STAT);
|
||
|
wait_cr_ack();
|
||
|
writel(CR_READ_ENABLE, SATA_PHY_ASIC_DATA);
|
||
|
wait_cr_ack();
|
||
|
return readl(SATA_PHY_ASIC_STAT);
|
||
|
}
|
||
|
|
||
|
static void write_cr(unsigned short data, unsigned short address)
|
||
|
{
|
||
|
writel(address, SATA_PHY_ASIC_STAT);
|
||
|
wait_cr_ack();
|
||
|
writel((data | CR_CAP_DATA), SATA_PHY_ASIC_DATA);
|
||
|
wait_cr_ack();
|
||
|
writel(CR_WRITE_ENABLE, SATA_PHY_ASIC_DATA);
|
||
|
wait_cr_ack();
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
void workaround5458(void)
|
||
|
{
|
||
|
unsigned i;
|
||
|
|
||
|
for (i = 0; i < 2; i++) {
|
||
|
unsigned short rx_control = read_cr(0x201d + (i << 8));
|
||
|
rx_control &= ~(PH_GAIN_MASK | FR_GAIN_MASK);
|
||
|
rx_control |= PH_GAIN << PH_GAIN_OFFSET;
|
||
|
rx_control |= FR_GAIN << FR_GAIN_OFFSET;
|
||
|
rx_control |= USE_INT_SETTING;
|
||
|
write_cr(rx_control, 0x201d + (i << 8));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
int ide_preinit(void)
|
||
|
{
|
||
|
int num_disks_found = 0;
|
||
|
|
||
|
/* Initialise records of which disks are present to all present */
|
||
|
int i;
|
||
|
for (i = 0; i < CONFIG_SYS_IDE_MAXDEVICE; i++) {
|
||
|
disk_present[i] = 1;
|
||
|
}
|
||
|
|
||
|
/* Block reset SATA and DMA cores */
|
||
|
reset_block(SYS_CTRL_RST_SATA, 1);
|
||
|
reset_block(SYS_CTRL_RST_SATA_LINK, 1);
|
||
|
reset_block(SYS_CTRL_RST_SATA_PHY, 1);
|
||
|
reset_block(SYS_CTRL_RST_SGDMA, 1);
|
||
|
|
||
|
/* Enable clocks to SATA and DMA cores */
|
||
|
enable_clock(SYS_CTRL_CLK_SATA);
|
||
|
enable_clock(SYS_CTRL_CLK_DMA);
|
||
|
|
||
|
udelay(5000);
|
||
|
reset_block(SYS_CTRL_RST_SATA_PHY, 0);
|
||
|
udelay(50);
|
||
|
reset_block(SYS_CTRL_RST_SATA, 0);
|
||
|
reset_block(SYS_CTRL_RST_SATA_LINK, 0);
|
||
|
udelay(50);
|
||
|
reset_block(SYS_CTRL_RST_SGDMA, 0);
|
||
|
udelay(100);
|
||
|
/* Apply the Synopsis SATA PHY workarounds */
|
||
|
workaround5458();
|
||
|
udelay(10000);
|
||
|
|
||
|
/* disable and clear core interrupts */
|
||
|
*((unsigned long*) SATA_HOST_REGS_BASE + SATA_INT_ENABLE_CLR_OFF) =
|
||
|
~0UL;
|
||
|
*((unsigned long*) SATA_HOST_REGS_BASE + SATA_INT_CLR_OFF) = ~0UL;
|
||
|
|
||
|
int device;
|
||
|
for (device = 0; device < CONFIG_SYS_IDE_MAXDEVICE; device++) {
|
||
|
int found = 0;
|
||
|
int retries = 1;
|
||
|
|
||
|
/* Disable SATA interrupts */
|
||
|
*(sata_regs_base[device] + SATA_INT_ENABLE_CLR_OFF) = ~0UL;
|
||
|
|
||
|
/* Clear any pending SATA interrupts */
|
||
|
*(sata_regs_base[device] + SATA_INT_CLR_OFF) = ~0UL;
|
||
|
|
||
|
do {
|
||
|
/* clear sector count register for FIS detection */
|
||
|
ide_outb(device, ATA_PORT_NSECT, 0);
|
||
|
|
||
|
/* Get the PHY working */
|
||
|
if (!phy_reset(device)) {
|
||
|
printf("SATA PHY not ready for device %d\n",
|
||
|
device);
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
if (!wait_FIS(device)) {
|
||
|
printf("No FIS received from device %d\n",
|
||
|
device);
|
||
|
} else {
|
||
|
if ((scr_read(device, SATA_SCR_STATUS) & 0xf)
|
||
|
== 0x3) {
|
||
|
if (wait_not_busy(device, 30)) {
|
||
|
printf("Timed out of wait for SATA device %d to have BUSY clear\n",
|
||
|
device);
|
||
|
} else {
|
||
|
++num_disks_found;
|
||
|
found = 1;
|
||
|
}
|
||
|
} else {
|
||
|
printf("No SATA device %d found, PHY status = 0x%08x\n",
|
||
|
device,
|
||
|
scr_read(
|
||
|
device,
|
||
|
SATA_SCR_STATUS));
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
} while (retries--);
|
||
|
|
||
|
/* Record whether disk is present, so won't attempt to access it later */
|
||
|
disk_present[device] = found;
|
||
|
}
|
||
|
|
||
|
/* post disk detection clean-up */
|
||
|
for (device = 0; device < CONFIG_SYS_IDE_MAXDEVICE; device++) {
|
||
|
if (disk_present[device]) {
|
||
|
/* set as ata-5 (28-bit) */
|
||
|
*(sata_regs_base[device] + SATA_DRIVE_CONTROL_OFF) =
|
||
|
0UL;
|
||
|
|
||
|
/* clear phy/link errors */
|
||
|
scr_write(device, SATA_SCR_ERROR, ~0);
|
||
|
|
||
|
/* clear host errors */
|
||
|
*(sata_regs_base[device] + SATA_CONTROL_OFF) |=
|
||
|
SATA_SCTL_CLR_ERR;
|
||
|
|
||
|
/* clear interrupt register as this clears the error bit in the IDE
|
||
|
status register */
|
||
|
*(sata_regs_base[device] + SATA_INT_CLR_OFF) = ~0UL;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return !num_disks_found;
|
||
|
}
|
||
|
|