реклама на сайте
подробности

 
 
2 страниц V  < 1 2  
Reply to this topicStart new topic
> SD/MMC, CSD и CID
zltigo
сообщение Dec 30 2007, 21:29
Сообщение #16


Гуру
******

Группа: Свой
Сообщений: 13 372
Регистрация: 27-11-04
Из: Riga, Latvia
Пользователь №: 1 244



Цитата(hsx_Vlad @ Dec 30 2007, 23:22) *
Вот теперь такой вопрос, как определить какой байт смотреть?

Повторяю последнй раз - дождитесь готовности и получите нормальные 16 байт, без произвольного количества незначаших байт перед информационным блоком.


--------------------
Feci, quod potui, faciant meliora potentes
Go to the top of the page
 
+Quote Post
hsx_Vlad
сообщение Dec 30 2007, 21:40
Сообщение #17


Участник
*

Группа: Новичок
Сообщений: 17
Регистрация: 15-08-07
Пользователь №: 29 819



насколько я знаю, дма генерит приерывание только в случае ошибки или вычитывания заданного количества байт sad.gif, т.е пока все требуемые байты не будут готовы, обработка дальше не пойдет

Сообщение отредактировал hsx_Vlad - Dec 30 2007, 21:42
Go to the top of the page
 
+Quote Post
zltigo
сообщение Dec 30 2007, 21:48
Сообщение #18


Гуру
******

Группа: Свой
Сообщений: 13 372
Регистрация: 27-11-04
Из: Riga, Latvia
Пользователь №: 1 244



Цитата(hsx_Vlad @ Dec 30 2007, 23:40) *
насколько я знаю, дма ...

Еще не поняли? Повторяю - тупо использовать DMA для вычитывания блока нельзя, ибо размер его априори не известен.
Читаем по байтику, и только получив правильный token вычитываем блок желаемого размера.
Код
//---------------------------------------------------------------------------
// Receive a data packet from MMC
//---------------------------------------------------------------------------
static int get_datablock( BYTE *dest, int count )
{
BYTE token;
ulong timer = xGetTimeout_ms( SD_RD_TIME_OUT );    // Wait for data packet in timeout
    do
    {    token = mmc_sendbyte( DUMMY_BYTE );
    
    }
    while( (token == 0xFF )&&( xIsTimeout( timer ) == FALSE ) );
    
    if( token != DATA_TOKEN_17_18_24 )
        return( FALSE );                // If not valid data token, retutn with error
    
    mmc_receiveblock( dest, count );      
    mmc_sendbyte( 0xFF );                 // Discard CRC
    mmc_sendbyte( 0xFF );                // Discard CRC

    return( TRUE );                        // Return with success
}


--------------------
Feci, quod potui, faciant meliora potentes
Go to the top of the page
 
+Quote Post
hsx_Vlad
сообщение Dec 30 2007, 22:03
Сообщение #19


Участник
*

Группа: Новичок
Сообщений: 17
Регистрация: 15-08-07
Пользователь №: 29 819



а что есть DATA_TOKEN_17_18_24?
Go to the top of the page
 
+Quote Post
zltigo
сообщение Dec 30 2007, 22:14
Сообщение #20


Гуру
******

Группа: Свой
Сообщений: 13 372
Регистрация: 27-11-04
Из: Riga, Latvia
Пользователь №: 1 244



Цитата(hsx_Vlad @ Dec 31 2007, 00:03) *
а что есть DATA_TOKEN_17_18_24?

А документ почитать?
Код
// Time ote definition for SD Read Time out ~100msec, Write Time out ~250ms
#define SD_RD_TIME_OUT      100
#define SD_WR_TIME_OUT        250
//
#define R1_OK                  0x00
#define R1_IDLE_STATE          0x01
#define R1_ERASE_RST           0x02
#define R1_ILLEGAL_CMD         0x04
#define R1_CRC_ERROR           0x08
#define R1_ERASE_ERROR         0x10
#define R1_ADD_ERROR           0x20
#define R1_PARAM_ERROR         0x40
#define R1_NOTVALID         0x80
#define R1_NOTRESPONCE         0xFF

#define DUMMY_BYTE             0xFF

#define DATA_TOKEN_17         0xFE
#define DATA_TOKEN_18         DATA_TOKEN_17
#define DATA_TOKEN_24         DATA_TOKEN_17
#define    DATA_TOKEN_17_18_24 DATA_TOKEN_17

#define DATA_TOKEN_25         0xFC
#define DATA_TOKEN_25_STOP  0xFD
#define DATA_TOKEN_ERROR    0x1F

#define DATA_RESP_MASK        0x1F
#define DATA_RESP_ACCEPT    0x05
#define DATA_RESP_CRC_ERR    0x0B
#define DATA_RESP_WR_ERR      0x0D


--------------------
Feci, quod potui, faciant meliora potentes
Go to the top of the page
 
+Quote Post
hsx_Vlad
сообщение Dec 31 2007, 11:05
Сообщение #21


Участник
*

Группа: Новичок
Сообщений: 17
Регистрация: 15-08-07
Пользователь №: 29 819



Спасибо, но проблема чуть не в этом. Часть вычитывающая блок у меня работает нормально.

static DWORD MMC_read_block (u8 cmd, DWORD arg, u8 *buf, DWORD len)
{
DWORD res = CARD_ERROR;
DWORD i, k, indx;
u8 constval = 0xFF;
k = 0;
i = 0;
res = CARD_ERROR;
sd_att = SD_MAX_ATT;
do
{
SD_SELECT_CARD;
cmd_buf[0] = 0xFF;
cmd_buf[1] = 0xFF;
cmd_buf[2] = 0xFF;
cmd_buf[3] = 0xFF;
cmd_buf[4] = cmd;
cmd_buf[5] = (arg >> 24);
cmd_buf[6] = (arg >> 16);
cmd_buf[7] = (arg >> 8);
cmd_buf[8] = (arg);
cmd_buf[9] = 0xFF;
cmd_buf[10] = 0xFF;
cmd_buf[11] = 0xFF;
cmd_buf[12] = 0xFF; // Cb_end, Cb_transfer_err функции изменяющие значение ssp_flgs
ssp_flgs = 0; // Устанавливается в прерывании DMA в значения SSP_DONE или SSP_TX_ERR
Start_dma_read_from_SSPSD(cmd_buf, 13, Cb_end, Cb_transfer_err); // После этого чтения в буфере
Start_dma_write_to_SSPSD(cmd_buf, 13, 0, Cb_transfer_err); // 11 байт должен стать равным 0 если все нормально
while ( (ssp_flgs&(SSP_DONE|SSP_TX_ERR))==0 ); // количество данных, которые необходимо прочитать извесны 13, а становиться либо 11 либо 12
if(ssp_flgs&SSP_TX_ERR ) goto exit__;
if ( ((cmd_buf[11] & 0x80)==0) || ((cmd_buf[12] & 0x80)==0))

{
if(cmd_buf[11]!=0 && cmd_buf[12]!=0)
{
res = cmd_buf[i];
goto exit__;
}
sd_ratt = SD_MAX_RATT;
do
{
ssp_flgs = 0;
Start_dma_read_from_SSPSD(cmd_buf, READ_WAIT_LEN, Cb_end, Cb_transfer_err);
Start_dma_write_const_to_SSPSD(&constval, READ_WAIT_LEN, 0, Cb_transfer_err);
while((ssp_flgs&(SSP_DONE|SSP_TX_ERR))==0 );
if(ssp_flgs&SSP_TX_ERR ) goto exit__;
for ( i=0;i<READ_WAIT_LEN;i++ )
{
if ( cmd_buf[i] == 0xFE )
{
k = i + 1;
break;
}
}
if ( cmd_buf[i] == 0xFE ) break;
sd_ratt--;
if ( sd_ratt==0 ) goto exit_att__;
}
while ( 1 );
indx = 0;
for ( i=k;i<READ_WAIT_LEN;i++ )
{
buf[indx++] = cmd_buf[i];
if ( indx==len ) break;
}
if ( indx!=len )
{
ssp_flgs = 0;
Start_dma_read_from_SSPSD(&buf[indx], len - indx, Cb_end, Cb_transfer_err);
Start_dma_write_const_to_SSPSD(&constval, len - indx, 0, Cb_transfer_err);
while ( (ssp_flgs&(SSP_DONE|SSP_TX_ERR))==0 );
if ( ssp_flgs&SSP_TX_ERR) goto exit__;
}
ssp_flgs = 0;
Start_dma_read_from_SSPSD(cmd_buf, 3, Cb_end, Cb_transfer_err);
Start_dma_write_const_to_SSPSD(&constval, 3, 0, Cb_transfer_err);
while ( (ssp_flgs&(SSP_DONE|SSP_TX_ERR))==0 );
if ( ssp_flgs&SSP_TX_ERR ) goto exit__;
res = CARD_OK;
goto exit__;
}
exit_att__: SD_UNSELECT_CARD;
sd_att--;
}
while ( sd_att!=0 );
exit__: SD_UNSELECT_CARD;
return res;
}

Сообщение отредактировал hsx_Vlad - Dec 31 2007, 11:08
Go to the top of the page
 
+Quote Post
zltigo
сообщение Dec 31 2007, 12:26
Сообщение #22


Гуру
******

Группа: Свой
Сообщений: 13 372
Регистрация: 27-11-04
Из: Riga, Latvia
Пользователь №: 1 244



Цитата(hsx_Vlad @ Dec 31 2007, 13:05) *
Часть вычитывающая блок у меня работает нормально.

Нет.

В качестве Новогоднего подарка еще два кусочка исходников опять таки с таймаутами и готовностями smile.gif кои с ..... ну, скажем, упорством достойным лучшего применения Вы пытаетесь игнорировать.

Код
//---------------------------------------------------------------------------
// Send a command to MMC
//---------------------------------------------------------------------------
static BYTE send_cmd( BYTE cmd, DWORD arg )
{
BYTE nn;
BYTE res;

    if( wait_ready() )
        return( 0x8F );

    // Send command packet
    mmc_sendbyte( cmd );                       // Command
    
    mmc_sendbyte( arg >> 24 );        // Argument[31..24]
    mmc_sendbyte( arg >> 16 );        // Argument[23..16]
    mmc_sendbyte( arg >> 8  );      // Argument[15...8]
    mmc_sendbyte( arg       );        // Argument[7....0]
    
    if( cmd == CMD0_RESET )
        nn = 0x95;            // CRC for CMD0  (0)
    else if( cmd == CMD8_IF_V2 )
        nn = 0x87;            // CRC for CMD8 (0x1AA)
    else
        nn = 0x00;
                    
    mmc_sendbyte( nn );

    // Receive command response
    if( cmd == CMD12_STOP_TX )
        mmc_sendbyte( 0xFF );            // Skip a stuff byte when stop reading
    
    // Wait for a valid response in timeout of 10 attempts    
    nn = 10;                                
    do
    {    res = mmc_sendbyte( 0xFF );
    
    }
    while( ( res & R1_NOTVALID )&&( --nn ) );

    return( res );                        // Return with the response value
}

//---------------------------------------------------------------------------
// Wait for card ready  
//---------------------------------------------------------------------------
static int wait_ready(void)
{
// Wait for ready in timeout
ulong timer = xGetTimeout_ms( SD_READY_TIME_OUT );            
BYTE ret;    
    mmc_sendbyte( 0xFF );
    
    do
    {   ret = mmc_sendbyte( 0xFF );
        if( ret == 0xFF )
            return( 0 );                
    }
    while( xIsTimeout( timer ) == FALSE );
    return( 1 );
}


Moderator:
Воздержитесь от постов длиных (воспользуйтесь возможностями приложений) невнятных неформатированных (ознакомьтесь с возможностями тэгов) кусков исходных текстов.


--------------------
Feci, quod potui, faciant meliora potentes
Go to the top of the page
 
+Quote Post
hsx_Vlad
сообщение Dec 31 2007, 13:21
Сообщение #23


Участник
*

Группа: Новичок
Сообщений: 17
Регистрация: 15-08-07
Пользователь №: 29 819



Спасибо, вот это меня как раз и интерисовало, получается что место нахождения ответа на команду тоже плавающее.
Go to the top of the page
 
+Quote Post
zltigo
сообщение Dec 31 2007, 14:16
Сообщение #24


Гуру
******

Группа: Свой
Сообщений: 13 372
Регистрация: 27-11-04
Из: Riga, Latvia
Пользователь №: 1 244



Цитата(hsx_Vlad @ Dec 31 2007, 15:21) *
получается что место нахождения ответа на команду тоже плавающее.

Естественно и документировано. Естественно, потому, что в карточке свой контроллер и самая разнообразная по быстродействию начинка. Попытки тупо всегда и везде использовать блочный DMA приводят к кривизне. Используйте DMA в эквиваленте mmc_receiveblock() - сие будет разумно и правильно. Остальное по байтикам. Даже притягивание за уши посылки, например, одним блоком команды прведет только к дополнительным потерям времени на формирование буфера и затратам памяти для самого буфера.


--------------------
Feci, quod potui, faciant meliora potentes
Go to the top of the page
 
+Quote Post

2 страниц V  < 1 2
Reply to this topicStart new topic
1 чел. читают эту тему (гостей: 1, скрытых пользователей: 0)
Пользователей: 0

 


RSS Текстовая версия Сейчас: 18th June 2025 - 02:31
Рейтинг@Mail.ru


Страница сгенерированна за 0.0145 секунд с 7
ELECTRONIX ©2004-2016