/****************************************************************************** * * Copyright (C) 2013 - 2019 Xilinx, Inc. All rights reserved. * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN * THE SOFTWARE. * * * ******************************************************************************/ /*****************************************************************************/ /** * * @file xsdps.h * @addtogroup sdps_v3_8 * @{ * @details * * This file contains the implementation of XSdPs driver. * This driver is used initialize read from and write to the SD card. * Features such as switching bus width to 4-bit and switching to high speed, * changing clock frequency, block size etc. are supported. * SD 2.0 uses 1/4 bus width and speeds of 25/50KHz. Initialization, however * is done using 1-bit bus width and 400KHz clock frequency. * SD commands are classified as broadcast and addressed. Commands can be * those with response only (using only command line) or * response + data (using command and data lines). * Only one command can be sent at a time. During a data transfer however, * when dsta lines are in use, certain commands (which use only the command * line) can be sent, most often to obtain status. * This driver does not support multi card slots at present. * * Initialization: * This includes initialization on the host controller side to select * clock frequency, bus power and default transfer related parameters. * The default voltage is 3.3V. * On the SD card side, the initialization and identification state diagram is * implemented. This resets the card, gives it a unique address/ID and * identifies key card related specifications. * * Data transfer: * The SD card is put in transfer state to read from or write to it. * The default block size is 512 bytes and if supported, * default bus width is 4-bit and bus speed is High speed. * The read and write functions are implemented in polled mode using ADMA2. * * At any point, when key parameters such as block size or * clock/speed or bus width are modified, this driver takes care of * maintaining the same selection on host and card. * All error bits in host controller are monitored by the driver and in the * event one of them is set, driver will clear the interrupt status and * communicate failure to the upper layer. * * File system use: * This driver can be used with xilffs library to read and write files to SD. * (Please refer to procedure in diskio.c). The file system read/write example * in polled mode can used for reference. * * There is no example for using SD driver without file system at present. * However, the driver can be used without the file system. The glue layer * in filesystem can be used as reference for the same. The block count * passed to the read/write function in one call is limited by the ADMA2 * descriptor table and hence care will have to be taken to call read/write * API's in a loop for large file sizes. * * Interrupt mode is not supported because it offers no improvement when used * with file system. * * eMMC support: * SD driver supports SD and eMMC based on the "enable MMC" parameter in SDK. * The features of eMMC supported by the driver will depend on those supported * by the host controller. The current driver supports read/write on eMMC card * using 4-bit and high speed mode currently. * * Features not supported include - card write protect, password setting, * lock/unlock, interrupts, SDMA mode, programmed I/O mode and * 64-bit addressed ADMA2, erase/pre-erase commands. * *
 * MODIFICATION HISTORY:
 *
 * Ver   Who    Date     Changes
 * ----- ---    -------- -----------------------------------------------
 * 1.00a hk/sg  10/17/13 Initial release
 * 2.0   hk      03/07/14 Version number revised.
 * 2.1   hk     04/18/14 Increase sleep for eMMC switch command.
 *                       Add sleep for microblaze designs. CR# 781117.
 * 2.2   hk     07/28/14 Make changes to enable use of data cache.
 * 2.3   sk     09/23/14 Send command for relative card address
 *                       when re-initialization is done.CR# 819614.
 *						Use XSdPs_Change_ClkFreq API whenever changing
 *						clock.CR# 816586.
 * 2.4	sk	   12/04/14 Added support for micro SD without
 *                      WP/CD. CR# 810655.
 *						Checked for DAT Inhibit mask instead of CMD
 *                      Inhibit mask in Cmd Transfer API.
 *						Added Support for SD Card v1.0
 * 2.5  sg		07/09/15 Added SD 3.0 features
 *       kvn     07/15/15 Modified the code according to MISRAC-2012.
 * 2.6   sk     10/12/15 Added support for SD card v1.0 CR# 840601.
 * 2.7   sk     11/24/15 Considered the slot type befoe checking CD/WP pins.
 *       sk     12/10/15 Added support for MMC cards.
 *              01/08/16 Added workaround for issue in auto tuning mode
 *                       of SDR50, SDR104 and HS200.
 *       sk     02/16/16 Corrected the Tuning logic.
 *       sk     03/01/16 Removed Bus Width check for eMMC. CR# 938311.
 * 2.8   sk     04/20/16 Added new workaround for auto tuning.
 *              05/03/16 Standard Speed for SD to 19MHz in ZynqMPSoC. CR#951024
 * 3.0   sk     06/09/16 Added support for mkfs to calculate sector count.
 *       sk     07/16/16 Added support for UHS modes.
 *       sk     07/07/16 Used usleep API for both arm and microblaze.
 *       sk     07/16/16 Added Tap delays accordingly to different SD/eMMC
 *                       operating modes.
 *       sk     08/13/16 Removed sleep.h from xsdps.h as a temporary fix for
 *                       CR#956899.
 * 3.1   mi     09/07/16 Removed compilation warnings with extra compiler flags.
 *       sk     10/13/16 Reduced the delay during power cycle to 1ms as per spec
 *       sk     10/19/16 Used emmc_hwreset pin to reset eMMC.
 *       sk     11/07/16 Enable Rst_n bit in ext_csd reg if not enabled.
 *       sk     11/16/16 Issue DLL reset at 31 iteration to load new zero value.
 * 3.2   sk     11/30/16 Modified the voltage switching sequence as per spec.
 *       sk     02/01/17 Added HSD and DDR mode support for eMMC.
 *       sk     02/01/17 Consider bus width parameter from design for switching
 *       vns    02/09/17 Added ARMA53_32 support for ZynqMP CR#968397
 *       sk     03/20/17 Add support for EL1 non-secure mode.
 * 3.3   mn     05/17/17 Add support for 64bit DMA addressing
 *  mn     08/07/17 Modify driver to support 64-bit DMA in arm64 only
 *       mn     08/17/17 Enabled CCI support for A53 by adding cache coherency
 *                       information.
 *       mn     09/06/17 Resolved compilation errors with IAR toolchain
 * 3.6   mn     08/01/18 Add support for using 64Bit DMA with 32-Bit Processor
 * 3.7   mn     02/01/19 Add support for idling of SDIO
 * 3.8   mn     04/12/19 Modified TapDelay code for supporting ZynqMP and Versal
 *       mn     09/17/19 Modified ADMA handling API for 32bit and 64bit addresses
 *
 * 
* ******************************************************************************/ #ifndef SDPS_H_ #define SDPS_H_ #ifdef __cplusplus extern "C" { #endif #include "xil_printf.h" #include "xil_cache.h" #include "xstatus.h" #include "xsdps_hw.h" #include "xplatform_info.h" #include /************************** Constant Definitions *****************************/ #define XSDPS_CT_ERROR 0x2L /**< Command timeout flag */ #define MAX_TUNING_COUNT 40U /**< Maximum Tuning count */ #define MAX_TIMEOUT 0x1FFFFFFFU /**< Maximum Timeout */ /**************************** Type Definitions *******************************/ typedef void (* XSdPs_ConfigTap) ( u32 Bank, u32 DeviceId, u32 CardType ); /** * This typedef contains configuration information for the device. */ typedef struct { u16 DeviceId; /**< Unique ID of device */ u32 BaseAddress; /**< Base address of the device */ u32 InputClockHz; /**< Input clock frequency */ u32 CardDetect; /**< Card Detect */ u32 WriteProtect; /**< Write Protect */ u32 BusWidth; /**< Bus Width */ u32 BankNumber; /**< MIO Bank selection for SD */ u32 HasEMIO; /**< If SD is connected to EMIO */ u8 IsCacheCoherent; /**< If SD is Cache Coherent or not */ } XSdPs_Config; /* ADMA2 32-Bit descriptor table */ typedef struct { u16 Attribute; /**< Attributes of descriptor */ u16 Length; /**< Length of current dma transfer */ u32 Address; /**< Address of current dma transfer */ #ifdef __ICCARM__ #pragma data_alignment = 32 } XSdPs_Adma2Descriptor32; #else } __attribute__( ( __packed__ ) ) XSdPs_Adma2Descriptor32; #endif /* ADMA2 64-Bit descriptor table */ typedef struct { u16 Attribute; /**< Attributes of descriptor */ u16 Length; /**< Length of current dma transfer */ u64 Address; /**< Address of current dma transfer */ #ifdef __ICCARM__ #pragma data_alignment = 32 } XSdPs_Adma2Descriptor64; #else } __attribute__( ( __packed__ ) ) XSdPs_Adma2Descriptor64; #endif /** * The XSdPs driver instance data. The user is required to allocate a * variable of this type for every SD device in the system. A pointer * to a variable of this type is then passed to the driver API functions. */ typedef struct { XSdPs_Config Config; /**< Configuration structure */ u32 IsReady; /**< Device is initialized and ready */ u32 Host_Caps; /**< Capabilities of host controller */ u32 Host_CapsExt; /**< Extended Capabilities */ u32 HCS; /**< High capacity support in card */ u8 CardType; /**< Type of card - SD/MMC/eMMC */ u8 Card_Version; /**< Card version */ u8 HC_Version; /**< Host controller version */ u8 BusWidth; /**< Current operating bus width */ u32 BusSpeed; /**< Current operating bus speed */ u8 Switch1v8; /**< 1.8V Switch support */ u32 CardID[ 4 ]; /**< Card ID Register */ u32 RelCardAddr; /**< Relative Card Address */ u32 CardSpecData[ 4 ]; /**< Card Specific Data Register */ u32 SectorCount; /**< Sector Count */ u32 SdCardConfig; /**< Sd Card Configuration Register */ u32 Mode; /**< Bus Speed Mode */ u32 OTapDelay; /**< Output Tap Delay */ u32 ITapDelay; /**< Input Tap Delay */ u64 Dma64BitAddr; /**< 64 Bit DMA Address */ } XSdPs; /***************** Macros (Inline Functions) Definitions *********************/ /************************** Function Prototypes ******************************/ XSdPs_Config * XSdPs_LookupConfig( u16 DeviceId ); s32 XSdPs_CfgInitialize( XSdPs * InstancePtr, XSdPs_Config * ConfigPtr, u32 EffectiveAddr ); s32 XSdPs_SdCardInitialize( XSdPs * InstancePtr ); s32 XSdPs_ReadPolled( XSdPs * InstancePtr, u32 Arg, u32 BlkCnt, u8 * Buff ); s32 XSdPs_WritePolled( XSdPs * InstancePtr, u32 Arg, u32 BlkCnt, const u8 * Buff ); s32 XSdPs_SetBlkSize( XSdPs * InstancePtr, u16 BlkSize ); s32 XSdPs_Select_Card( XSdPs * InstancePtr ); s32 XSdPs_Change_ClkFreq( XSdPs * InstancePtr, u32 SelFreq ); s32 XSdPs_Change_BusWidth( XSdPs * InstancePtr ); s32 XSdPs_Change_BusSpeed( XSdPs * InstancePtr ); s32 XSdPs_Get_BusWidth( XSdPs * InstancePtr, u8 * ReadBuff ); s32 XSdPs_Get_BusSpeed( XSdPs * InstancePtr, u8 * ReadBuff ); s32 XSdPs_Get_Status( XSdPs * InstancePtr, u8 * SdStatReg ); s32 XSdPs_Pullup( XSdPs * InstancePtr ); s32 XSdPs_MmcCardInitialize( XSdPs * InstancePtr ); s32 XSdPs_CardInitialize( XSdPs * InstancePtr ); s32 XSdPs_Get_Mmc_ExtCsd( XSdPs * InstancePtr, u8 * ReadBuff ); s32 XSdPs_Set_Mmc_ExtCsd( XSdPs * InstancePtr, u32 Arg ); void XSdPs_Idle( XSdPs * InstancePtr ); #if defined( ARMR5 ) || defined( __aarch64__ ) || defined( ARMA53_32 ) || defined( __MICROBLAZE__ ) void XSdPs_Identify_UhsMode( XSdPs * InstancePtr, u8 * ReadBuff ); void XSdPs_ddr50_tapdelay( u32 Bank, u32 DeviceId, u32 CardType ); void XSdPs_hsd_sdr25_tapdelay( u32 Bank, u32 DeviceId, u32 CardType ); void XSdPs_sdr104_hs200_tapdelay( u32 Bank, u32 DeviceId, u32 CardType ); #endif /* if defined( ARMR5 ) || defined( __aarch64__ ) || defined( ARMA53_32 ) || defined( __MICROBLAZE__ ) */ /* Make sure that the stored status is in ulSDInterruptStatus[] is cleared. * ulMask are the status bits that should come high: either CC or CC|TC. */ void XSdPs_ClearInterrupt( XSdPs * InstancePtr, uint32_t ulMask ); /* Wait for an interrupt and return the 32 bits of the status register. * A return value of 0 means: time-out. */ u32 XSdPs_WaitInterrupt( XSdPs * InstancePtr, u32 ulMask, u32 ulWait ); s32 XSdPs_Wait_For( XSdPs * InstancePtr, u32 Mask, u32 Wait ); u32 XSdPs_ReadStatus( XSdPs * InstancePtr ); u32 UNSTUFF_BITS( u32 * ulResponse, int iFirst, int iSize ); #ifdef __cplusplus } #endif #endif /* SDPS_H_ */ /** @} */