rt-thread/bsp/phytium/libraries/standalone/drivers/sata/fsata/fsata.h

284 lines
9.4 KiB
C
Raw Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*
* Copyright : (C) 2022 Phytium Information Technology, Inc.
* All Rights Reserved.
*
* This program is OPEN SOURCE software: you can redistribute it and/or modify it
* under the terms of the Phytium Public License as published by the Phytium Technology Co.,Ltd,
* either version 1.0 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 Phytium Public License for more details.
*
*
* FilePath: fsata.h
* Date: 2022-02-10 14:55:11
* LastEditTime: 2022-02-18 09:05:24
* Description:  This file is for sata ctrl function definition
*
* Modify History:
* Ver   Who        Date         Changes
* ----- ------     --------    --------------------------------------
* 1.0 wangxiaodong 2022/2/10 first release
* 1.1 wangxiaodong 2022/9/9 improve functions
* 1.2 wangxiaodong 2022/10/21 improve functions
*/
#ifndef FSATA_H
#define FSATA_H
#include "ftypes.h"
#include "ferror_code.h"
#include "fkernel.h"
#ifdef __cplusplus
extern "C"
{
#endif
#define FSATA_SUCCESS FT_SUCCESS
#define FSATA_ERR_INVAILD_PARAMETER FT_MAKE_ERRCODE(ErrModBsp, ErrBspSata, 1)
#define FSATA_ERR_TIMEOUT FT_MAKE_ERRCODE(ErrModBsp, ErrBspSata, 2)
#define FSATA_ERR_OPERATION FT_MAKE_ERRCODE(ErrModBsp, ErrBspSata, 3)
#define FSATA_UNKNOWN_DEVICE FT_MAKE_ERRCODE(ErrModBsp, ErrBspSata, 4)
/************************** Constant Definitions *****************************/
#define FSATA_AHCI_MAX_PORTS 32
#define FSATA_AHCI_MAX_PRD_ENTRIES 16
#define MAX_DATA_BYTE_COUNT SZ_4M
#define FSATA_AHCI_CMD_LIST_HEADER_SIZE 0x20
#define FSATA_AHCI_CMD_LIST_HEADER_NUM 32
#define FSATA_AHCI_RX_FIS_SZ 0x100
#define FSATA_AHCI_CMD_TABLE_HEADER_SIZE 0x80
#define FSATA_AHCI_PRTD_ITEM_SIZE 0x10
#define FSATA_AHCI_PRTD_ITEM_NUM 0x40 /*set 64 item, hardware max is 64K */
#define FSATA_AHCI_CMD_TABLE_SIZE (FSATA_AHCI_CMD_TABLE_HEADER_SIZE + (FSATA_AHCI_PRTD_ITEM_NUM * FSATA_AHCI_PRTD_ITEM_SIZE))
#define FSATA_AHCI_PORT_PRIV_DMA_SZ (FSATA_AHCI_CMD_LIST_HEADER_SIZE * FSATA_AHCI_CMD_LIST_HEADER_NUM + \
FSATA_AHCI_CMD_TABLE_SIZE + FSATA_AHCI_RX_FIS_SZ)
#define FSATA_AHCI_CMD_ATAPI BIT(5)
#define FSATA_AHCI_CMD_WRITE BIT(6)
#define FSATA_AHCI_CMD_PREFETCH BIT(7)
#define FSATA_AHCI_CMD_RESET BIT(8)
#define FSATA_AHCI_CMD_CLR_BUSY BIT(10)
#define FSATA_ID_LBA48_SECTORS 100
#define FSATA_ID_LBA_SECTORS 60
#define FSATA_ID_ATA_DEVICE BIT(15) /* IDENTIFY DEVICE word 0, if ATA device */
#define FSATA_ID_COMPLETE BIT(2) /* IDENTIFY DEVICE word 0, if the content of the IDENTIFY DEVICE data is incomplete */
#define FSATA_ID_FW_REV 23 /* firmware revision position */
#define FSATA_ID_PROD 27 /* Model number position */
#define FSATA_ID_WORDS 256 /* IDENTIFY DEVICE data length */
enum
{
FSATA_FIS_REG_HOST_TO_DEVICE = 0x27,
FSATA_FIS_REG_DEVICE_TO_HOST = 0x34,
FSATA_FIS_DMA_SETUP = 0x41
};
#define FSATA_FIS_REG_HOST_TO_DEVICE_C BIT(7) /* update of the command register */
#define FSATA_CMD_EXT_DEVICE BIT(6) /* command device byte requirement */
enum
{
FSATA_CMD_READ_EXT = 0x25,
FSATA_CMD_WRITE_EXT = 0x35,
FSATA_CMD_IDENTIFY_DEVICE = 0xEC,
FSATA_CMD_FPDMA_READ = 0x60,
FSATA_CMD_FPDMA_WRITE = 0x61
};
#define FSATA_BUSY BIT(7) /* BSY status bit */
#define FSATA_SECT_SIZE 512 /* sata sector size */
#define FSATA_BLK_VEN_SIZE 40 /* device vendor string size */
#define FSATA_BLK_PRD_SIZE 20 /* device product number size */
#define FSATA_BLK_REV_SIZE 8 /* firmware revision size */
#define FSATA_DEV_TYPE_UNKNOWN 0xff /* not connected */
#define FSATA_DEV_TYPE_HARDDISK 0x00 /* harddisk */
#define FSATA_IF_TYPE_UNKNOWN 0xff
#define FSATA_IF_TYPE_SCSI 0x00
enum
{
FSATA_TYPE_PCIE = 0,
FSATA_TYPE_CONTROLLER = 1
};
/************************** Variable Definitions *****************************/
/***************** Macros (Inline Functions) Definitions *********************/
/* Number of User Addressable Logical Sectors lba28 */
#define FSATA_ID_U32(id,n) \
(((u32) (id)[(n) + 1] << 16) | ((u32) (id)[(n)]))
/* Number of User Addressable Logical Sectors lba48 */
#define FSATA_ID_U64(id,n) \
( ((u64) (id)[(n) + 3] << 48) | \
((u64) (id)[(n) + 2] << 32) | \
((u64) (id)[(n) + 1] << 16) | \
((u64) (id)[(n) + 0]) )
/* if sata id is support lba */
#define FSataIdHasLba(id) ((id)[49] & BIT(9))
/* if sata id is support lba48 */
static inline int FSataIdHasLba48(const u16 *id)
{
if ((id[83] & 0xC000) != 0x4000)
{
return 0;
}
if (!FSATA_ID_U64(id, 100))
{
return 0;
}
return id[83] & BIT(10);
}
/**************************** Type Definitions *******************************/
typedef void (*FSataIrqCallBack)(void *args);
/* sata info */
typedef struct
{
unsigned char if_type; /* type of the interface */
unsigned char type; /* device type */
char vendor[FSATA_BLK_VEN_SIZE + 1]; /* device vendor string */
char product[FSATA_BLK_PRD_SIZE + 1]; /* device product number */
char revision[FSATA_BLK_REV_SIZE + 1]; /* firmware revision */
unsigned long lba; /* number of blocks */
unsigned long lba512; /* number of blocks of 512 bytes */
unsigned long blksz; /* block size */
} FSataInfo;
/* Received FIS Structure */
typedef struct __attribute__((__packed__))
{
u8 dma_setup_fis[28];
u8 reserved0[4];
u8 pio_setup_fis[20];
u8 reserved1[12];
u8 d2h_register_fis[20];
u8 reserved2[4];
u8 set_device_bits_fis[8];
u8 unknown_fis[64];
u8 reserved3[96];
}
FSataAhciRecvFis;
/* command list structure - command header */
typedef struct
{
u32 description_info;/* DW 0 Description Information */
u32 status; /* DW 1 - Command Status */
u32 tbl_addr; /* DW 2 Command Table Base Address */
u32 tbl_addr_hi; /* DW 3 Command Table Base Address Upper */
u32 reserved[4];
} FSataAhciCommandList;
/* command table - PRDT */
typedef struct
{
u32 addr_low; /* DW 0 Data Base Address */
u32 addr_high; /* DW 1 Data Base Address Upper */
u32 reserved; /* DW 2 Reserved */
u32 data_byte; /* DW 3 Description Information */
} FSataAhciCommandTablePrdt;
/* ahci port information structure */
typedef struct
{
uintptr port_base_addr; /* port base address */
FSataAhciCommandList *cmd_list; /* Command List structure, will include cmd_tbl's address */
FSataAhciRecvFis *rx_fis; /* Received FIS Structure */
uintptr cmd_tbl_base_addr; /* command table addr, also the command table's first part */
FSataAhciCommandTablePrdt *cmd_tbl_prdt;/* command table's second part , cmd_tbl + cmd_tbl_prdt = command table*/
FSataInfo dev_info;
} FSataAhciPorts;
typedef struct
{
u32 instance_id; /* Device instance id */
uintptr base_addr; /* sata base address */
char *instance_name; /* instance name */
u32 irq_num; /* Irq number */
} FSataConfig; /* sata config */
typedef struct
{
FSataConfig config;
u32 is_ready;
FSataAhciPorts port[FSATA_AHCI_MAX_PORTS];
u16 *ataid[FSATA_AHCI_MAX_PORTS];
u32 n_ports; /* maximum number of ports supported by the ahci, Number of Ports (NP)*/
u32 port_map; /* each bit indicate port can be used, If a bit is set to 1, the corresponding port is available for software to use. */
u32 link_port_map; /* each bit indicate port linkup sata device */
u32 private_data; /* each bit indicate port sata achi started */
FSataIrqCallBack fsata_dhrs_cb; /* device-to-host register fis interrupt */
void *dhrs_args;
FSataIrqCallBack fsata_pss_cb; /* pio setup fis interrupt */
void *pss_args;
FSataIrqCallBack fsata_dss_cb; /* dma setup fis interrupt */
void *dss_args;
FSataIrqCallBack fsata_sdbs_cb; /* set device bits interrupt */
void *sdbs_args;
FSataIrqCallBack fsata_pcs_cb; /* port connect change status interrupt */
void *pcs_args;
volatile u8 dhrs_flag;
volatile u8 sdb_flag;
} FSataCtrl;
/************************** Function Prototypes ******************************/
/* sata config init */
const FSataConfig *FSataLookupConfig(u32 instance_id, u8 type);
/* initialize sata ctrl */
FError FSataCfgInitialize(FSataCtrl *instance_p, const FSataConfig *input_config_p);
/* deinitialize sata ctrl */
void FSataCfgDeInitialize(FSataCtrl *pctrl);
/* read sata info */
FError FSataAhciReadInfo(FSataCtrl *instance_p, u8 port);
/* init ahci */
FError FSataAhciInit(FSataCtrl *instance_p);
/* init ahci port */
FError FSataAhciPortStart(FSataCtrl *instance_p, u8 port, uintptr mem);
/* read or write sata data */
FError FSataReadWrite(FSataCtrl *instance_p, u8 port, u32 start,
u16 blk_cnt, u8 *buffer, boolean is_ncq, boolean is_write);
/* sata all irq handler entry */
void FSataIrqHandler(s32 vector, void *param);
/* set specific sata irq function entry */
FError FSataSetHandler(FSataCtrl *instance_p, u32 handler_type,
void *func_pointer, void *call_back_ref);
/* set sata irq mask */
void FSataIrqEnable(FSataCtrl *instance_p, u32 int_mask);
#ifdef __cplusplus
}
#endif
#endif