|
Не могу передать пакеты хосту по усб, at91sam7s256 |
|
|
|
Dec 29 2007, 07:06
|
Местный
  
Группа: Участник
Сообщений: 206
Регистрация: 12-10-06
Из: ufa
Пользователь №: 21 241

|
Использую стандартную прошивку AT91SAM7S256-USART_USB_SAM7S-IAR4_31A-1_00 для связи контроллера с усб, только переписал дескрипторы, чтобы работала с моим драйвером и убрал код связанный с усарт. Нумерация проходит нормально. Writefile тоже работает. Пакеты благополучно доставляются моему девайсу. Не могу реализовать передачу пакетов от девайса хосту. Readfile возвращает ошибку. В стандартной прошивке в файле cdc_enumerate.c я нашел функцию AT91F_UDP_IsConfigured(AT91PS_CDC pCdc), которая, как я понял отвечает за обработку источников прерываний isr. Туда я добавил свой код, кот обрабатывает прерывания от ер1 и ер2 (ер1 - in, ep2 - out). Вот этот код: Код //*---------------------------------------------------------------------------- //* \fn AT91F_UDP_IsConfigured //* \brief Test if the device is configured and handle enumeration //*---------------------------------------------------------------------------- static uchar AT91F_UDP_IsConfigured(AT91PS_CDC pCdc) { AT91PS_UDP pUDP = pCdc->pUdp; AT91_REG isr = pUDP->UDP_ISR;
if (isr & AT91C_UDP_ENDBUSRES) { pUDP->UDP_ICR = AT91C_UDP_ENDBUSRES; // reset all endpoints pUDP->UDP_RSTEP = (unsigned int)-1; pUDP->UDP_RSTEP = 0; // Enable the function pUDP->UDP_FADDR = AT91C_UDP_FEN; // Configure endpoint 0 pUDP->UDP_CSR[0] = (AT91C_UDP_EPEDS | AT91C_UDP_EPTYPE_CTRL); } else if (isr & AT91C_UDP_EPINT0) { pUDP->UDP_ICR = AT91C_UDP_EPINT0; AT91F_CDC_Enumerate(pCdc); } else if (isr & AT91C_UDP_EPINT1){ pUDP->UDP_ICR = AT91C_UDP_EPINT1; pUDP->UDP_CSR[1]=0; //подтверждает, что хост принял данные } else if (isr & AT91C_UDP_EPINT2){ pUDP->UDP_ICR = AT91C_UDP_EPINT2; AT91F_CDC_ReadData(pCdc); } return pCdc->currentConfiguration; } Т.е. при получении данных от хоста возникает прерывание от ер2 и вызывается функция AT91F_CDC_ReadData(pCdc). Ее код приведен ниже: Код extern struct _AT91S_CDC pCDC; static char adrRT[64]; AT91PS_UDP pUDP = pCdc->pUdp; pCDC.Read(&pCDC, adrRT,64); uint nbytes = AT91F_UDP_Read(pCdc, adrRT, 64); pUDP->UDP_CSR[2]=pUDP->UDP_CSR[2]&0x7FFFFFD;/*сброс бита RX_DATA_BK0 - подтвердение того, что данные из FIFO считаны*/ static char CWord[64]; CWord[1] = nbytes; pCDC.Write(&pCDC,CWord,64); Т.е. я считываю данные из буфера, а потом пытаюсь передать количество полученных байт обратно на хост, но они не передаются, т.к. device monitoring studio не показывает принятых байт от моего девайса. Привожу код функции main() из файла main.c Код // Enable User Reset and set its minimal assertion to 960 us AT91C_BASE_RSTC->RSTC_RMR = AT91C_RSTC_URSTEN | (0x4<<8) | (unsigned int)(0xA5<<24); // Init USB device AT91F_USB_Open(); // Init USB device // Wait for the end of enumeration while (!pCDC.IsConfigured(&pCDC)); while(1); Как правильно воспользоваться этой стандартной прошивкой, чтобы считывать и записывать данные в конечные точки? Может я свой код не туда добавил, вроде кроме как в AT91F_UDP_IsConfigured(AT91PS_CDC pCdc) и некуда, она же прерывания обрабатывает или я не прав?
|
|
|
|
|
Jan 3 2008, 16:19
|
Группа: Новичок
Сообщений: 4
Регистрация: 19-07-07
Пользователь №: 29 255

|
Мне кажется, Вы вообще не стой стороны зашли. Почитайте про AIC. И переделайте прерывания .
Только вот тут одна проблема у меня схожая есть. Не возникают эти прерывания на энд-поинтах. А данные передаются. O_o
Функция чтения данных в строчке length = pCDC.Read(&pCDC, data, MSG_SIZE); Отправки в строчке pCDC.Write(&pCDC, data, length);
Но если пытаться посадить их на прерывания они не срабатывают.
Может, кто знает как их из main {...} в прерывания перенести?
Сообщение отредактировал DimitryB - Jan 3 2008, 16:21
|
|
|
|
|
Jan 3 2008, 17:28
|
Местный
  
Группа: Участник
Сообщений: 206
Регистрация: 12-10-06
Из: ufa
Пользователь №: 21 241

|
Цитата Может, кто знает как их из main {...} в прерывания перенести? Да меня это тоже интересует!
|
|
|
|
|
Jan 3 2008, 20:45
|
Группа: Новичок
Сообщений: 4
Регистрация: 19-07-07
Пользователь №: 29 255

|
Цитата(Bulat @ Jan 3 2008, 21:28)  Да меня это тоже интересует! Если вдруг узнаете сообщите плз admin[@]bulkin.info А для device monitoring studio не поделетесь крякой?
|
|
|
|
|
Jan 4 2008, 09:11
|

Местный
  
Группа: Свой
Сообщений: 387
Регистрация: 20-12-06
Из: Obninsk
Пользователь №: 23 719

|
Цитата(Bulat @ Dec 29 2007, 10:06)  Как правильно воспользоваться этой стандартной прошивкой, чтобы считывать и записывать данные в конечные точки? Может я свой код не туда добавил, вроде кроме как в AT91F_UDP_IsConfigured(AT91PS_CDC pCdc) и некуда, она же прерывания обрабатывает или я не прав? AT91F_UDP_IsConfigured не обрабатывает прерывания. Сконфигурируйте в main прерывания от EP0, что то типа этого: if (HID.IsConfigured(&HID)) { HID.pUdp->UDP_ICR = 0xff00; AT91F_UDP_DisableIt ( HID.pUdp, 0xFFFF); AT91F_UDP_EnableIt ( HID.pUdp, AT91C_UDP_EPINT2); AT91F_AIC_ConfigureIt ( AT91C_BASE_AIC, AT91C_ID_UDP, UDP_INTERRUPT_LEVEL,AT91C_AIC_SRCTYPE_INT_LEVEL_SENSITIVE, udp_c_irq_handler); AT91F_AIC_EnableIt (AT91C_BASE_AIC, AT91C_ID_UDP);} Обработка прерывания: __ramfunc __arm void udp_c_irq_handler(void){ AT91PS_UDP pUdp = HID.pUdp; u_int packetSize, nbBytesRcv = 0, currentReceiveBank = HID.currentRcvBank; pUdp->UDP_CSR[AT91C_EP_OUT] &= ~(AT91C_UDP_TXCOMP | AT91C_UDP_RXSETUP | AT91C_UDP_ISOERROR ); while (length_in) { if ( pUdp->UDP_CSR[AT91C_EP_OUT] & (AT91C_UDP_RX_DATA_BK0 | AT91C_UDP_RX_DATA_BK1) ) { packetSize = MIN(pUdp->UDP_CSR[AT91C_EP_OUT] >> 16, length_in); length_in -= packetSize; if (packetSize < AT91C_EP_OUT_SIZE) length_in = 0; while(packetSize--) data_in[nbBytesRcv++] = pUdp->UDP_FDR[AT91C_EP_OUT]; pUdp->UDP_CSR[AT91C_EP_OUT] &= ~(currentReceiveBank); if (currentReceiveBank == AT91C_UDP_RX_DATA_BK0) currentReceiveBank = AT91C_UDP_RX_DATA_BK1; else currentReceiveBank = AT91C_UDP_RX_DATA_BK0; } } // Write_ack(); //Если надо. HID.currentRcvBank = currentReceiveBank; }
|
|
|
|
|
Jan 6 2008, 16:24
|
Местный
  
Группа: Участник
Сообщений: 206
Регистрация: 12-10-06
Из: ufa
Пользователь №: 21 241

|
Цитата А для device monitoring studio не поделетесь крякой? К сожалению сам пользуюсь 14-дневной версией, вот на работу выйду и надо искать новый комп, чтобы там эту прогу ставить(
|
|
|
|
|
Jan 9 2008, 06:34
|
Местный
  
Группа: Участник
Сообщений: 206
Регистрация: 12-10-06
Из: ufa
Пользователь №: 21 241

|
Цитата(KAlex @ Jan 4 2008, 14:11)  AT91F_UDP_IsConfigured не обрабатывает прерывания. Сконфигурируйте в main прерывания от EP0, что то типа этого: if (HID.IsConfigured(&HID)) { HID.pUdp->UDP_ICR = 0xff00; AT91F_UDP_DisableIt ( HID.pUdp, 0xFFFF); AT91F_UDP_EnableIt ( HID.pUdp, AT91C_UDP_EPINT2); AT91F_AIC_ConfigureIt ( AT91C_BASE_AIC, AT91C_ID_UDP, UDP_INTERRUPT_LEVEL,AT91C_AIC_SRCTYPE_INT_LEVEL_SENSITIVE, udp_c_irq_handler); AT91F_AIC_EnableIt (AT91C_BASE_AIC, AT91C_ID_UDP);}
Обработка прерывания: __ramfunc __arm void udp_c_irq_handler(void){ AT91PS_UDP pUdp = HID.pUdp; u_int packetSize, nbBytesRcv = 0, currentReceiveBank = HID.currentRcvBank; pUdp->UDP_CSR[AT91C_EP_OUT] &= ~(AT91C_UDP_TXCOMP | AT91C_UDP_RXSETUP | AT91C_UDP_ISOERROR );
while (length_in) { if ( pUdp->UDP_CSR[AT91C_EP_OUT] & (AT91C_UDP_RX_DATA_BK0 | AT91C_UDP_RX_DATA_BK1) ) { packetSize = MIN(pUdp->UDP_CSR[AT91C_EP_OUT] >> 16, length_in); length_in -= packetSize; if (packetSize < AT91C_EP_OUT_SIZE) length_in = 0; while(packetSize--) data_in[nbBytesRcv++] = pUdp->UDP_FDR[AT91C_EP_OUT]; pUdp->UDP_CSR[AT91C_EP_OUT] &= ~(currentReceiveBank); if (currentReceiveBank == AT91C_UDP_RX_DATA_BK0) currentReceiveBank = AT91C_UDP_RX_DATA_BK1; else currentReceiveBank = AT91C_UDP_RX_DATA_BK0;
} } // Write_ack(); //Если надо. HID.currentRcvBank = currentReceiveBank; } Спасибо, но хотелось бы еще посмотреть на пример кода осуществляющий запись данных в конечную точку EP_IN, для передачи их хосту. Да и еще вопрос. В функции AT91F_AIC_ConfigureIt ( AT91C_BASE_AIC, AT91C_ID_UDP, UDP_INTERRUPT_LEVEL,AT91C_AIC_SRCTYPE_INT_LEVEL_SENSITIVE, udp_c_irq_handler) не определены параметры UDP_INTERRUPT_LEVEL и AT91C_AIC_SRCTYPE_INT_LEVEL_SENSITIVE. Где они определяются?
|
|
|
|
|
Jan 10 2008, 10:46
|

Местный
  
Группа: Свой
Сообщений: 387
Регистрация: 20-12-06
Из: Obninsk
Пользователь №: 23 719

|
Цитата(Bulat @ Jan 9 2008, 09:34)  Спасибо, но хотелось бы еще посмотреть на пример кода осуществляющий запись данных в конечную точку EP_IN, для передачи их хосту. __ramfunc __arm u_int AT91F_UDP_Write(AT91PS_HID pCdc, char *pData, u_int length){ AT91PS_UDP pUdp = pCdc->pUdp; u_int cpt = 0; if (length) while ( (pUdp->UDP_CSR[AT91C_EP_IN] & AT91C_UDP_TXPKTRDY) && !(pUdp->UDP_ISR & AT91C_UDP_RXSUSP) ); // Send the first packet cpt = MIN(length, FTDI_EP_IN_SIZE); length -= cpt; while (cpt--) pUdp->UDP_FDR[AT91C_EP_IN] = *pData++; pUdp->UDP_CSR[AT91C_EP_IN] |= AT91C_UDP_TXPKTRDY; while (length) { // Fill the second bank cpt = MIN(length, FTDI_EP_IN_SIZE); length -= cpt; while ( (pUdp->UDP_CSR[AT91C_EP_IN] & AT91C_UDP_TXPKTRDY) && !(pUdp->UDP_ISR & AT91C_UDP_RXSUSP) ); if ( pUdp->UDP_ISR & AT91C_UDP_RXSUSP ) return length; while (cpt--) pUdp->UDP_FDR[AT91C_EP_IN] = *pData++; // Wait for the the first bank to be sent while ( !(pUdp->UDP_CSR[AT91C_EP_IN] & AT91C_UDP_TXCOMP) ) if ( !AT91F_UDP_IsConfigured(pCdc) ) return length; pUdp->UDP_CSR[AT91C_EP_IN] &= ~(AT91C_UDP_TXCOMP); while (pUdp->UDP_CSR[AT91C_EP_IN] & AT91C_UDP_TXCOMP); pUdp->UDP_CSR[AT91C_EP_IN] |= AT91C_UDP_TXPKTRDY; } // Wait for the end of transfer while ( !(pUdp->UDP_CSR[AT91C_EP_IN] & AT91C_UDP_TXCOMP) ){ if ( !AT91F_UDP_IsConfigured(pCdc) ) return length; }; pUdp->UDP_CSR[AT91C_EP_IN] &= ~(AT91C_UDP_TXCOMP); while (pUdp->UDP_CSR[AT91C_EP_IN] & AT91C_UDP_TXCOMP); return length; } Для ускоренной передачи пакетов не более 64 байта: __ramfunc __arm void AT91F_UDP_Write64(AT91PS_HID pCdc, char *pData, char length){ AT91PS_UDP pUdp = pCdc->pUdp; while ( (pUdp->UDP_CSR[AT91C_EP_IN] & AT91C_UDP_TXPKTRDY) && !(pUdp->UDP_ISR & AT91C_UDP_RXSUSP) ); pUdp->UDP_CSR[AT91C_EP_IN] &= ~(AT91C_UDP_TXCOMP); while (length--) pUdp->UDP_FDR[AT91C_EP_IN] = *pData++; pUdp->UDP_CSR[AT91C_EP_IN] |= AT91C_UDP_TXPKTRDY; } Цитата(Bulat @ Jan 9 2008, 09:34)  Да и еще вопрос. В функции AT91F_AIC_ConfigureIt ( AT91C_BASE_AIC, AT91C_ID_UDP, UDP_INTERRUPT_LEVEL,AT91C_AIC_SRCTYPE_INT_LEVEL_SENSITIVE, udp_c_irq_handler) не определены параметры UDP_INTERRUPT_LEVEL и AT91C_AIC_SRCTYPE_INT_LEVEL_SENSITIVE. Где они определяются? Делается один раз в main: if (HID.IsConfigured(&HID)) { HID.pUdp->UDP_ICR = 0xff00; AT91F_UDP_DisableIt ( HID.pUdp, 0xFFFF); AT91F_UDP_EnableIt ( HID.pUdp, AT91C_UDP_EPINT2); AT91F_AIC_ConfigureIt ( AT91C_BASE_AIC, AT91C_ID_UDP, UDP_INTERRUPT_LEVEL,AT91C_AIC_SRCTYPE_INT_LEVEL_SENSITIVE, udp_c_irq_handler); AT91F_AIC_EnableIt (AT91C_BASE_AIC, AT91C_ID_UDP); }
|
|
|
|
|
Jan 11 2008, 08:32
|
Местный
  
Группа: Участник
Сообщений: 206
Регистрация: 12-10-06
Из: ufa
Пользователь №: 21 241

|
Цитата Да и еще вопрос. В функции AT91F_AIC_ConfigureIt ( AT91C_BASE_AIC, AT91C_ID_UDP, UDP_INTERRUPT_LEVEL,AT91C_AIC_SRCTYPE_INT_LEVEL_SENSITIVE, udp_c_irq_handler) не определены параметры UDP_INTERRUPT_LEVEL и AT91C_AIC_SRCTYPE_INT_LEVEL_SENSITIVE. Где они определяются?QuoteEndQuoteEEndДелается один раз в main: if (HID.IsConfigured(&HID)) { HID.pUdp->UDP_ICR = 0xff00; AT91F_UDP_DisableIt ( HID.pUdp, 0xFFFF); AT91F_UDP_EnableIt ( HID.pUdp, AT91C_UDP_EPINT2); AT91F_AIC_ConfigureIt ( AT91C_BASE_AIC, AT91C_ID_UDP, UDP_INTERRUPT_LEVEL,AT91C_AIC_SRCTYPE_INT_LEVEL_SENSITIVE, udp_c_irq_handler); AT91F_AIC_EnableIt (AT91C_BASE_AIC, AT91C_ID_UDP);} Я именно так и сделал, но компилятор пишет: Error[Pe020]: identifier "UDP_INTERRUPT_LEVEL" is undefined Error[Pe020]: identifier "AT91C_AIC_SRCTYPE_INT_LEVEL_SENSITIVE" is undefined Может просто вместо этих полей конкретные числа поставить? Вот только какие?
|
|
|
|
|
Jan 11 2008, 08:53
|

Местный
  
Группа: Свой
Сообщений: 387
Регистрация: 20-12-06
Из: Obninsk
Пользователь №: 23 719

|
Цитата(Bulat @ Jan 11 2008, 11:32)  Я именно так и сделал, но компилятор пишет: Error[Pe020]: identifier "UDP_INTERRUPT_LEVEL" is undefined Error[Pe020]: identifier "AT91C_AIC_SRCTYPE_INT_LEVEL_SENSITIVE" is undefined
Может просто вместо этих полей конкретные числа поставить? Вот только какие? AT91C_AIC_SRCTYPE_INT_LEVEL_SENSITIVE определен в AT91SAM7Sххх.h UDP_INTERRUPT_LEVEL определяешь сам.
|
|
|
|
|
Jan 14 2008, 06:22
|
Местный
  
Группа: Участник
Сообщений: 206
Регистрация: 12-10-06
Из: ufa
Пользователь №: 21 241

|
Может я неправильно вызываю функцию записи UDP_Write() в обработчике прерывания? (пытаюсь передать 64 байт). На всякий случай привожу код с main: Код #define MSG_SIZE 1000 #define MIN(a, b) (((a) < (b)) ? (a) : (b)) #define AT91C_EP_IN_SIZE 0x40 #define UDP_INTERRUPT_LEVEL 0x7 //*-------------------------------------------------------------------------------------- //* Function Name : main //* Object : //*-------------------------------------------------------------------------------------- int main ( void ) { __ramfunc __arm void udp_c_irq_handler(void); // Enable User Reset and set its minimal assertion to 960 us AT91C_BASE_RSTC->RSTC_RMR = AT91C_RSTC_URSTEN | (0x4<<8) | (unsigned int)(0xA5<<24); // Init USB device AT91F_USB_Open(); // Init USB device // Wait for the end of enumeration while (!pCDC.IsConfigured(&pCDC)); pCDC.pUdp->UDP_ICR = 0xff00; AT91F_UDP_DisableIt ( pCDC.pUdp, 0xFFFF); AT91F_UDP_EnableIt (pCDC.pUdp, AT91C_UDP_EPINT2); AT91F_AIC_ConfigureIt ( AT91C_BASE_AIC, AT91C_ID_UDP, UDP_INTERRUPT_LEVEL,AT91C_AIC_SRCTYPE_INT_HIGH_LEVEL, udp_c_irq_handler); AT91F_AIC_EnableIt (AT91C_BASE_AIC, AT91C_ID_UDP); }
//udp_c_irq_handler __ramfunc __arm void udp_c_irq_handler(void) { char buff_r[64]; static char buff_w[64]; pCDC.Write = UDP_Write; pCDC.Read = UDP_Read; unsigned int ret_val = pCDC.Read(&pCDC, buff_r,64); for(int i=0;i<64;i++) { buff_w[i] = i; } pCDC.Write(&pCDC, buff_w,64); }
//*---------------------------------------------------------------------------- //* \fn UDP_Read //* \brief Read available data from Endpoint OUT 2 //*---------------------------------------------------------------------------- __ramfunc __arm unsigned int UDP_Read(AT91PS_CDC pCdc, char *data_in, unsigned int length_in) { AT91PS_UDP pUdp = pCDC.pUdp; unsigned int packetSize, nbBytesRcv = 0, currentReceiveBank = pCDC.currentRcvBank; pUdp->UDP_CSR[AT91C_EP_OUT] &= ~(AT91C_UDP_TXCOMP | AT91C_UDP_RXSETUP | AT91C_UDP_ISOERROR ); while (length_in) { if ( pUdp->UDP_CSR[AT91C_EP_OUT] & (AT91C_UDP_RX_DATA_BK0 | AT91C_UDP_RX_DATA_BK1) ) { packetSize = MIN(pUdp->UDP_CSR[AT91C_EP_OUT] >> 16, length_in); length_in -= packetSize; if (packetSize < AT91C_EP_OUT_SIZE) length_in = 0; while(packetSize--) data_in[nbBytesRcv++] = pUdp->UDP_FDR[AT91C_EP_OUT]; pUdp->UDP_CSR[AT91C_EP_OUT] &= ~(currentReceiveBank); if (currentReceiveBank == AT91C_UDP_RX_DATA_BK0) currentReceiveBank = AT91C_UDP_RX_DATA_BK1; else currentReceiveBank = AT91C_UDP_RX_DATA_BK0; } } pCDC.currentRcvBank = currentReceiveBank; return nbBytesRcv; }
//*---------------------------------------------------------------------------- //* \fn UDP_Write //* \brief Send through endpoint 1 //*---------------------------------------------------------------------------- __ramfunc __arm unsigned int UDP_Write(AT91PS_CDC pCdc, const char *pData, unsigned int length) { AT91PS_UDP pUdp = pCdc->pUdp; unsigned int cpt = 0; if (length) while ( (pUdp->UDP_CSR[AT91C_EP_IN] & AT91C_UDP_TXPKTRDY) && !(pUdp->UDP_ISR & AT91C_UDP_RXSUSP) ); // Send the first packet cpt = MIN(length, AT91C_EP_IN_SIZE); length -= cpt; while (cpt--) pUdp->UDP_FDR[AT91C_EP_IN] = *pData++; pUdp->UDP_CSR[AT91C_EP_IN] |= AT91C_UDP_TXPKTRDY; while (length) { // Fill the second bank cpt = MIN(length, AT91C_EP_IN_SIZE); length -= cpt; while ( (pUdp->UDP_CSR[AT91C_EP_IN] & AT91C_UDP_TXPKTRDY) && !(pUdp->UDP_ISR & AT91C_UDP_RXSUSP) ); if ( pUdp->UDP_ISR & AT91C_UDP_RXSUSP ) return length; while (cpt--) pUdp->UDP_FDR[AT91C_EP_IN] = *pData++; // Wait for the the first bank to be sent while ( !(pUdp->UDP_CSR[AT91C_EP_IN] & AT91C_UDP_TXCOMP) ) //if ( !AT91F_UDP_IsConfigured(pCdc) ) //return length; pUdp->UDP_CSR[AT91C_EP_IN] &= ~(AT91C_UDP_TXCOMP); while (pUdp->UDP_CSR[AT91C_EP_IN] & AT91C_UDP_TXCOMP); pUdp->UDP_CSR[AT91C_EP_IN] |= AT91C_UDP_TXPKTRDY; } // Wait for the end of transfer while ( !(pUdp->UDP_CSR[AT91C_EP_IN] & AT91C_UDP_TXCOMP) ){ //if ( !AT91F_UDP_IsConfigured(pCdc) ) //return length; }; pUdp->UDP_CSR[AT91C_EP_IN] &= ~(AT91C_UDP_TXCOMP); while (pUdp->UDP_CSR[AT91C_EP_IN] & AT91C_UDP_TXCOMP);
return length; }
|
|
|
|
|
Jan 15 2008, 11:06
|
Местный
  
Группа: Участник
Сообщений: 206
Регистрация: 12-10-06
Из: ufa
Пользователь №: 21 241

|
Помоему даже обработчик прерываний udp_c_irq_handler не вызывается... Вроде прерывания инициализировал так, как мне тут объяснили. Еще раз привожу функцию main и функциюобработчика прерваний. Посмотрите, пжста, почему у меня прерывания не вызываются от EP_OUT? Код #define MSG_SIZE 1000 #define MIN(a, b) (((a) < (b)) ? (a) : (b)) #define AT91C_EP_IN_SIZE 0x40 #define UDP_INTERRUPT_LEVEL 0x7 //*-------------------------------------------------------------------------------------- //* Function Name : main //* Object : //*-------------------------------------------------------------------------------------- int main ( void ) { __ramfunc __arm void udp_c_irq_handler(void); // Enable User Reset and set its minimal assertion to 960 us AT91C_BASE_RSTC->RSTC_RMR = AT91C_RSTC_URSTEN | (0x4<<8) | (unsigned int)(0xA5<<24); // Init USB device AT91F_USB_Open(); // Init USB device // Wait for the end of enumeration while (!pCDC.IsConfigured(&pCDC)); pCDC.pUdp->UDP_ICR = 0xff00; AT91F_UDP_DisableIt ( pCDC.pUdp, 0xFFFF); AT91F_UDP_EnableIt (pCDC.pUdp, AT91C_UDP_EPINT2); AT91F_AIC_ConfigureIt ( AT91C_BASE_AIC, AT91C_ID_UDP, UDP_INTERRUPT_LEVEL,AT91C_AIC_SRCTYPE_INT_HIGH_LEVEL, udp_c_irq_handler); AT91F_AIC_EnableIt (AT91C_BASE_AIC, AT91C_ID_UDP); }
//udp_c_irq_handler __ramfunc __arm void udp_c_irq_handler(void) { char buff_r[64]; static char buff_w[64]; pCDC.Write = UDP_Write; pCDC.Read = UDP_Read; pCDC.Read(&pCDC, buff_r,64); for(int i=0;i<64;i++) { buff_w[i] = i; } pCDC.Write(&pCDC, buff_w,64); } Заранее благодарен!
|
|
|
|
|
Jan 15 2008, 11:17
|

Гуру
     
Группа: Модераторы
Сообщений: 8 455
Регистрация: 15-05-06
Из: Рига, Латвия
Пользователь №: 17 095

|
Цитата(Bulat @ Jan 15 2008, 13:06)  Помоему даже обработчик прерываний udp_c_irq_handler не вызывается... А где сам обработчик IRQ? Что-то вроде Код #pragma vector = 0x18 __irq __arm void IRQ_Switch() { void (*Handler)() = (void(*)())AT91C_BASE_AIC->AIC_IVR; Handler(); AT91C_BASE_AIC->AIC_EOICR = 0; // Reset AIC logic } Где в main() глобальное разрешение разрешение прерываний ( __enable_interrupt(); )? хотя, возможно вы это просто не показали, тогда извиняюсь.
--------------------
На любой вопрос даю любой ответ"Write code that is guaranteed to work, not code that doesn’t seem to break" ( C++ FAQ)
|
|
|
|
|
Jan 16 2008, 05:09
|
Местный
  
Группа: Участник
Сообщений: 206
Регистрация: 12-10-06
Из: ufa
Пользователь №: 21 241

|
Цитата А где сам обработчик IRQ? Что-то вроде c1ec1#pragma vector = 0x18__irq __arm void IRQ_Switch() { void (*Handler)() = (void(*)())AT91C_BASE_AIC->AIC_IVR; Handler(); AT91C_BASE_AIC->AIC_EOICR = 0; // Reset AIC logic}c2ec2Где в main() глобальное разрешение разрешение прерываний ( __enable_interrupt(); )?хотя, возможно вы это просто не показали, тогда извиняюсь. Если я добавляю обработчик IRQ, девайс вообще перестает определяться! Я же за основу взял готовый пример USB-USART вот и пытаюсь его под себя переделать. Вот как выглядит код с обработчиком IRQ, с main и самим хандлером: Код #pragma vector = 0x18 __irq __arm void IRQ_Switch() { void (*udp_c_irq_handler)() = (void(*)())AT91C_BASE_AIC->AIC_IVR; udp_c_irq_handler(); AT91C_BASE_AIC->AIC_EOICR = 0; // Reset AIC logic }
int main ( void ) { __ramfunc __arm void udp_c_irq_handler(void); // Enable User Reset and set its minimal assertion to 960 us AT91C_BASE_RSTC->RSTC_RMR = AT91C_RSTC_URSTEN | (0x4<<8) | (unsigned int)(0xA5<<24); // Init USB device AT91F_USB_Open(); // Init USB device // Wait for the end of enumeration while(!pCDC.IsConfigured(&pCDC)); AT91C_BASE_AIC->AIC_IECR |= (1<<AT91C_ID_UDP); AT91C_BASE_AIC->AIC_IDCR = ~AT91C_BASE_AIC->AIC_IECR; pCDC.pUdp->UDP_ICR = 0xff00; AT91F_UDP_DisableIt ( pCDC.pUdp, 0xFFFF); AT91F_UDP_EnableIt (pCDC.pUdp, AT91C_UDP_EPINT2); AT91F_AIC_ConfigureIt ( AT91C_BASE_AIC, AT91C_ID_UDP, UDP_INTERRUPT_LEVEL,AT91C_AIC_SRCTYPE_HIGH_LEVEL, udp_c_irq_handler); AT91F_AIC_EnableIt (AT91C_BASE_AIC, AT91C_ID_UDP); }
//udp_c_irq_handler __ramfunc __arm void udp_c_irq_handler(void) { char buff_r[64]; static char buff_w[64]; //pCDC.Write = UDP_Write; //pCDC.Read = UDP_Read; unsigned int ret_val = UDP_Read(&pCDC, buff_r,64); for(int i=0;i<64;i++) { buff_w[i] = i; } UDP_Write(&pCDC, buff_w,64); } Т.е. добавив обработчик __irq __arm void IRQ_Switch() девайс перестает даже нумероваться! Почему?
|
|
|
|
1 чел. читают эту тему (гостей: 1, скрытых пользователей: 0)
Пользователей: 0
|
|
|