⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 at91rm9200hdd.c

📁 at91rm9200硬盘的接口代码
💻 C
字号:
//*----------------------------------------------------------------------------
//*         ATMEL Microcontroller Software Support  -  ROUSSET  -
//*----------------------------------------------------------------------------
//* The software is delivered "AS IS" without warranty or condition of any
//* kind, either express, implied or statutory. This includes without
//* limitation any warranty or condition with respect to merchantability or
//* fitness for any particular purpose, or against the infringements of
//* intellectual property rights of others.
//*----------------------------------------------------------------------------
//* File Name           : HDD.c
//* Object               : main application written in C
//* Creation            : GGi   12/11/2002
//*
//*----------------------------------------------------------------------------
#include "at91rm9200hdd.h"
#include "fs_port.h"
#include "fs_conf.h"


volatile char  *HDD_BASE	=  (char *)0x50c00000;
volatile short *HDD_BASE_16	= (short *)0x50c00000;

/*delay fct*/
void delay(long nb){
	while(nb--);
}


void FS_IDE_HW_X_BusyLedOn(FS_u32 Unit) {

	AT91F_PIO_CfgOutput(AT91C_BASE_PIOB, AT91C_PIO_PB25);
	AT91F_PIO_ClearOutput(AT91C_BASE_PIOB,AT91C_PIO_PB25);
    
}


/*********************************************************************
*
*             FS_IDE_HW_X_BusyLedOff
*
  Description:
  FS driver hardware layer function. Turn off busy led.

  Parameters:
  Unit        - Unit number.
 
  Return value:
  None.
*/

void FS_IDE_HW_X_BusyLedOff(FS_u32 Unit) {
	AT91F_PIO_CfgOutput(AT91C_BASE_PIOB, AT91C_PIO_PB25);
	AT91F_PIO_SetOutput(AT91C_BASE_PIOB,AT91C_PIO_PB25);
}

/*********************************************************************
*
*             FS_IDE_HW_X_DetectStatus
*
  Description:
  FS driver hardware layer function. Check if the device is present.

  Parameters:
  Unit        - Unit number.
 
  Return value:
  ==0         - Device is connected.
  !=0         - Device has not been found.
*/

char FS_IDE_HW_X_DetectStatus(FS_u32 Unit) {
	//AT91F_HDDOpen();
    return 0;
}



//*----------------------------------------------------------------------------
//* \fn    AT91F_HDDGetStatus
//* \brief wait the RDY status
//*----------------------------------------------------------------------------
 char AT91F_HDDGetStatus (void){
     return ( *(HDD_BASE + ATA_STATUS_OFFSET));
		
}

//*----------------------------------------------------------------------------
//* \fn    AT91F_HDD_StatusLow
//* \brief wait the corresponding bit of status falls
//*----------------------------------------------------------------------------
 int AT91F_HDD_StatusLow (int mask){
    struct _AT91S_SMC2 *PS_ebi = (struct _AT91S_SMC2 *) (AT91C_BASE_SMC2) ;

    char status = 0;
    int cpt = 0;
   
	/*swap to 8bits data bus width*/
	PS_ebi->SMC2_CSR[4] &= 	~(AT91C_SMC2_DBW_16|AT91C_SMC2_BAT);
	PS_ebi->SMC2_CSR[4] |= 	AT91C_SMC2_DBW_8;
   
    while (status && cpt++ < 100000){
			status = (AT91F_HDDGetStatus()) & mask;
	}

	/*back to 16*/
	PS_ebi->SMC2_CSR[4] &= 	~AT91C_SMC2_DBW_8;
	PS_ebi->SMC2_CSR[4] |= 	(AT91C_SMC2_DBW_16|AT91C_SMC2_BAT);
	
	if (cpt == 100000) return -1;
	return 0;
}

//*----------------------------------------------------------------------------
//* \fn    AT91F_HDD_StatusHigh
//* \brief wait the corresponding bit of status raise
//*----------------------------------------------------------------------------
 int AT91F_HDD_StatusHigh (int mask){
    struct _AT91S_SMC2 *PS_ebi = (struct _AT91S_SMC2 *) (AT91C_BASE_SMC2) ;

    char status = 0;
    int cpt = 0;
   
	/*swap to 8bits data bus width*/
	PS_ebi->SMC2_CSR[4] &= 	~(AT91C_SMC2_DBW_16|AT91C_SMC2_BAT);
	PS_ebi->SMC2_CSR[4] |= 	AT91C_SMC2_DBW_8;
   
    while (!status && cpt++ < 100000){
			status = (AT91F_HDDGetStatus()) & mask;
	}

	/*back to 16*/
	PS_ebi->SMC2_CSR[4] &= 	~AT91C_SMC2_DBW_8;
	PS_ebi->SMC2_CSR[4] |= 	(AT91C_SMC2_DBW_16|AT91C_SMC2_BAT);
	
	if (cpt == 100000) return -1;
	return 0;
}

//*----------------------------------------------------------------------------
//* \fn    AT91F_HDDWriteCommand
//* \brief Write a command to the HDD
//*----------------------------------------------------------------------------
int AT91F_HDDWriteCommand(char * buffer, int size, int offset){
	struct _AT91S_SMC2 *PS_ebi = (struct _AT91S_SMC2 *) (AT91C_BASE_SMC2) ;
	int i;
	
	if (AT91F_HDD_StatusLow (ATA_ST_BUSY) < 0) return -1;

	/*swap to 8bits data bus width*/
	PS_ebi->SMC2_CSR[4] &= 	~(AT91C_SMC2_DBW_16|AT91C_SMC2_BAT);
	PS_ebi->SMC2_CSR[4] |= 	AT91C_SMC2_DBW_8;
	
	for (i=0;i<size;i++){
			*(HDD_BASE + i + offset) = *buffer++;
			delay(10000);
			}

	/*back to 16*/
	PS_ebi->SMC2_CSR[4] &= 	~AT91C_SMC2_DBW_8;
	PS_ebi->SMC2_CSR[4] |= 	(AT91C_SMC2_DBW_16|AT91C_SMC2_BAT);
	
	return 0;
}

/*****************************************************************************************/
/*函数:AT91F_HDD_Read16                                                                  */
/*功能:从ATA数据寄存器读取指定个数的数据                                                 */
/*参数:*buffer数据缓冲区基地址,size 要读取的个数                                         */
/*****************************************************************************************/
int AT91F_HDD_Read16(unsigned short *buffer,int size){
	int cpt = 0;
	if (AT91F_HDD_StatusHigh (ATA_ST_DRQ) < 0) return -1;
	while ((cpt < size) && (cpt < BUF_SIZE)){
		*buffer++ =*(HDD_BASE_16);
		cpt ++;
	}
	return cpt;
}

//*----------------------------------------------------------------------------
//* \fn    AT91F_HDD_Write16
//* \brief Write to HDD
//*----------------------------------------------------------------------------
int AT91F_HDD_Write16(unsigned short *buffer,int size){
	int cpt = 0;
	if (AT91F_HDD_StatusHigh (ATA_ST_DRQ) < 0) return -1;


	while ((cpt < size) && (cpt < BUF_SIZE)){
		*(HDD_BASE_16 ) = *buffer++;
		cpt++;
	}

	return cpt;
}


/***************************** driver interface ******************************/

//*----------------------------------------------------------------------------
//* \fn    AT91F_HDDOpen
//* \brief Opens HDD device
//*----------------------------------------------------------------------------
void AT91F_HDDOpen(void){
     struct _AT91S_EBI *PS_ebi = (struct _AT91S_EBI *) (AT91C_BASE_EBI) ;
     struct _AT91S_SMC2 *PS_smc2 = (struct _AT91S_SMC2 *) (AT91C_BASE_SMC2) ;

	//CS configuration
	
    PS_smc2->SMC2_CSR[4]=(HDD_IO_RWH|HDD_IO_RWS|AT91C_SMC2_ACSS_STANDARD|AT91C_SMC2_BAT|AT91C_SMC2_DBW_16|\
    						HDD_IO_TDF|AT91C_SMC2_WSEN|HDD_IO_NWS);

	//PIO configuration

	// (EBI) Chip Select 4 is assigned to the Static Memory Controller and NCS4,NCS5 and NCS6
	// behave as defined by the SMC2.
	
	AT91F_PIO_CfgPeriph(AT91C_BASE_PIOC, AT91C_PC6_NWAIT, 0);
	//AT91F_PIO_CfgPeriph(AT91C_BASE_PIOC, AT91C_PC7_A23 , 0);
	AT91F_PIO_CfgPeriph(AT91C_BASE_PIOC, AT91C_PC9_A25_CFRNW, 0);
	AT91F_PIO_CfgPeriph(AT91C_BASE_PIOC, AT91C_PC10_NCS4_CFCS, 0);

	//IDE mode
	AT91F_PIO_CfgOutput(AT91C_BASE_PIOC, AT91C_PC11_NCS5_CFCE1);
	AT91F_PIO_ClearOutput(AT91C_BASE_PIOC, AT91C_PC11_NCS5_CFCE1);
	AT91F_PIO_CfgOutput(AT91C_BASE_PIOC, AT91C_PC12_NCS6_CFCE2);
	AT91F_PIO_SetOutput(AT91C_BASE_PIOC, AT91C_PC12_NCS6_CFCE2);

	//resets the HDD Card
	AT91F_PIO_CfgOutput(AT91C_BASE_PIOC, AT91C_PIO_PC5);
	AT91F_PIO_ClearOutput(AT91C_BASE_PIOC, AT91C_PIO_PC5);
	delay(100000);
	AT91F_PIO_SetOutput(AT91C_BASE_PIOC, AT91C_PIO_PC5);

	PS_ebi->EBI_CSA |= AT91C_EBI_CS4A_SMC_CompactFlash;

	delay(0x400000);
	

}

//*----------------------------------------------------------------------------
//* \fn    AT91F_HDDRead_ID
//* \brief Read the CF ID in 16bits databus width
//*----------------------------------------------------------------------------
 int AT91F_HDDRead_ID (unsigned short * buf){
 	char cmd[2] = {0,DRIVE_ID};

    if (AT91F_HDDWriteCommand(cmd,2,6) < 0) return -1;
    if (AT91F_HDD_Read16(buf,BUF_SIZE/2) != BUF_SIZE/2) return -1;
   	if (AT91F_HDD_StatusLow (ATA_ST_ERR) < 0) return (-1);

   	return 0;

}

//*----------------------------------------------------------------------------
//* \fn    AT91F_HDDErase
//* \brief erase a sector
//*----------------------------------------------------------------------------
int AT91F_HDDErase (int add){
 	char cmd[6];

 	/*CC*/cmd[5] = ERASE_SECTOR;
 	/*DR*/cmd[4] = (1 << 7) | (1 << 6) | (1 << 5) | ((add & 0xf000000) >> 24); //LBA = 1
 	/*CY*/cmd[3] = (add & 0xff0000) >> 16;
 	/*HD*/cmd[2] = (add & 0xff00) >> 8;
 	/*SN*/cmd[1] = (add & 0xff);
 	/*SC*/cmd[0] = 1;
    if(AT91F_HDDWriteCommand(cmd,6,2) < 0) return -1;
	if (AT91F_HDD_StatusLow (ATA_ST_ERR) < 0) return -1;

   	return 0;

}

//*----------------------------------------------------------------------------
//* \fn    HDDWrite
//* \brief write to the HDD in LBA mode
//*----------------------------------------------------------------------------
int AT91F_HDDWrite (unsigned short * buf , int add){
     char cmd[6];
 	/*CC*/cmd[5] = WRITE_SECTOR;
 	/*DR*/cmd[4] = (1 << 7) | (1 << 6) | (1 << 5) | ((add & 0xf000000) >> 24); //LBA = 1
 	/*CY*/cmd[3] = (add & 0xff0000) >> 16;
 	/*HD*/cmd[2] = (add & 0xff00) >> 8;
 	/*SN*/cmd[1] = (add & 0xff);
 	/*SC*/cmd[0] = 1;
    if (AT91F_HDDWriteCommand(cmd,6,2) < 0) return -1;
    if (AT91F_HDD_Write16(buf,BUF_SIZE/2) < BUF_SIZE/2) return -1;
   	if (AT91F_HDD_StatusLow (ATA_ST_ERR) < 0) return (-1);

   	return 0;

}
/*****************************************************************************************/
/*函数:AT91F_HDDRead                                                                     */
/*功能:从制定的扇区地址读一个扇区的数据                                                  */
/*参数:buf数据缓冲区基地址,add扇区地址                                                  */
/*****************************************************************************************/
int AT91F_HDDRead (unsigned short * buf , int add){
     char cmd[6];
 	 cmd[0] = 1;                       /*2=SC:扇区数  */
 	 cmd[1] = (add & 0xff);            /*3=SN:扇区号寄存器或LBA的A7~A0*/
 	 cmd[2] = (add & 0xff00) >> 8;     /*4=HD:柱面号寄存器(高字节)或LBA的A15~A8*/
 	 cmd[3] = (add & 0xff0000) >> 16;  /*5=CY:柱面号寄存器(高字节)或LBA的A23~A16*/
 	 cmd[4] = (1 << 7) | (1 << 6) | (1 << 5) | ((add & 0xf000000) >> 24);///*6=DR:驱动器或磁头寄存器LBA = 1*/
 	 cmd[5] = READ_SECTOR;             /*7=CC(LBA的A27~A24)命令寄存器*/

    if (AT91F_HDDWriteCommand(cmd,6,2) < 0) return -1;
    if (AT91F_HDD_Read16(buf,BUF_SIZE/2) < BUF_SIZE/2) return -1;
   	if (AT91F_HDD_StatusLow (ATA_ST_ERR) < 0) return (-1);

   	return 0;

}


//*----------------------------------------------------------------------------
//* \fn    HDDSleep
//* \brief put the HDD into sleep mode
//*----------------------------------------------------------------------------
int AT91F_HDDSleep (void){
 	char sleep[3];

 	sleep[0] = STAND_BY; 
 	sleep[1] = STAND_BY_IMM;
 	sleep[2] = SET_SLEEP_MODE;

    if (AT91F_HDDWriteCommand(&sleep[0],1,7) < 0) return -1;
    if (AT91F_HDDWriteCommand(&sleep[1],1,7) < 0) return -1;
    if (AT91F_HDDWriteCommand(&sleep[2],1,7) < 0) return -1;
   	if (AT91F_HDD_StatusLow (ATA_ST_ERR) < 0) return (-1);

   	return 0;

}


/******************************************************************************/

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -