pcan_cantact_another_one/Src/pcan_protocol.c

673 lines
14 KiB
C
Raw Normal View History

2020-06-03 04:33:18 -07:00
#include <assert.h>
#include <stdint.h>
#include <string.h>
#include "pcan_can.h"
#include "pcan_usb.h"
#include "pcan_packet.h"
#include "pcan_led.h"
#include "pcan_timestamp.h"
#include "punker.h"
static struct
{
2020-06-03 13:58:38 -07:00
uint8_t bus_active;
uint8_t ext_vcc_state;
uint8_t led_mode;
uint16_t last_timestamp_sync;
uint16_t last_flush;
struct
{
/* error handling related */
uint8_t err;
uint8_t ecc;
uint8_t rx_err;
uint8_t tx_err;
/* config */
uint8_t btr0;
uint8_t btr1;
uint8_t silient;
uint8_t loopback;
uint8_t err_mask;
uint8_t sja1000_mode;
uint32_t quartz_freq;
}
can;
uint8_t device_id;
uint32_t device_serial;
2020-06-03 04:33:18 -07:00
}
pcan_device =
{
2020-06-03 13:58:38 -07:00
.device_id = 0xFF,
.device_serial = 0xFFFFFFFF,
.can =
{
.quartz_freq = PCAN_USB_CRYSTAL_HZ,
.loopback = 0,
},
2020-06-03 04:33:18 -07:00
};
#pragma pack(push, 1)
#define PCAN_MAX_RECORD_SIZE (62)
typedef struct
{
/* type: 0x02 */
uint8_t header;
/* slot record count */
uint8_t rec_count;
/* record data */
uint8_t data[64-2];
}
PCAN_RECORD_ENTRY;
typedef struct
{
2020-06-03 13:58:38 -07:00
/* raw data array */
PCAN_RECORD_ENTRY buffer;
/* ts counter, need to provide correct timestamp correction ( 2 or 1 byte ) */
uint8_t ts_counter;
uint8_t pos;
2020-06-03 04:33:18 -07:00
}
PCAN_RECORD_BUFFER_EX;
#pragma pack(pop)
static PCAN_RECORD_BUFFER_EX pcan_records;
void pcan_record_buffer_init( PCAN_RECORD_BUFFER_EX *prec )
{
2020-06-03 13:58:38 -07:00
memset( prec, 0x00, sizeof( PCAN_RECORD_BUFFER_EX ) );
2020-06-03 04:33:18 -07:00
2020-06-03 13:58:38 -07:00
prec->buffer.header = 0x02;
prec->buffer.rec_count = 0x00;
2020-06-03 04:33:18 -07:00
}
void pcan_record_buffer_reset( PCAN_RECORD_BUFFER_EX *prec )
{
2020-06-03 13:58:38 -07:00
prec->buffer.rec_count = 0x00;
prec->pos = 0;
prec->ts_counter = 0;
2020-06-03 04:33:18 -07:00
}
uint16_t pcan_record_buffer_flush( PCAN_RECORD_BUFFER_EX *prec )
{
uint16_t flushed_size;
2020-06-03 13:58:38 -07:00
uint16_t ts_ticks = pcan_timestamp_ticks();
if( !prec->pos )
{
/* try to send ZLP, original 800 */
if( pcan_device.last_flush && (((uint16_t)(ts_ticks - pcan_device.last_flush)) >= PCAN_TICKS_FROM_US( 800 )) )
{
if( pcan_usb_send_data_buffer( "", 0 ) != 0xFFFF )
{
pcan_device.last_flush = 0;
}
}
return 0;
}
flushed_size = pcan_usb_send_data_buffer( &prec->buffer, sizeof( prec->buffer ) );
if( flushed_size == sizeof( prec->buffer ) )
{
pcan_device.last_flush = ts_ticks;
pcan_record_buffer_reset( prec );
}
2020-06-03 04:33:18 -07:00
return flushed_size;
}
/* request free slot for specific amount of data */
uint8_t *pcan_record_buffer_request( PCAN_RECORD_BUFFER_EX *prec, uint16_t size )
{
2020-06-03 13:58:38 -07:00
/* try to find free slot for specific data */
uint16_t pos = prec->pos;
if( ( pos + size ) > PCAN_MAX_RECORD_SIZE )
return (void*)0;
2020-06-03 04:33:18 -07:00
2020-06-03 13:58:38 -07:00
return &prec->buffer.data[pos];
2020-06-03 04:33:18 -07:00
}
void pcan_record_buffer_commit( PCAN_RECORD_BUFFER_EX *prec, uint16_t size )
{
2020-06-03 13:58:38 -07:00
prec->buffer.rec_count++;
prec->pos += size;
2020-06-03 04:33:18 -07:00
}
#if 0
static void pcan_bus_event( PCAN_RECORD_BUFFER_EX *prec, uint8_t flags )
{
2020-06-03 13:58:38 -07:00
uint8_t *ptr, sl, pos = 0;
ptr = pcan_record_buffer_request( prec, 6 );
if( !ptr )
return;
/* sl */
sl = PCAN_USB_STATUSLEN_INTERNAL | 0x03;
pack_u8( ptr, sl );
ptr += 1;
pos += 1;
/* function */
pack_u8( ptr, PCAN_USB_REC_BUSEVT );
ptr += 1;
pos += 1;
/* number 0x80, 0x00 */
pack_u8( ptr, flags );
ptr += 1;
pos += 1;
/* ecc */
pack_u8( ptr, pcan_device.can.ecc );
ptr += 1;
pos += 1;
/* rx error counter */
pack_u8( ptr, pcan_device.can.rx_err );
ptr += 1;
pos += 1;
/* tx error counter */
pack_u8( ptr, pcan_device.can.tx_err );
ptr += 1;
pos += 1;
/* add new record */
pcan_record_buffer_commit( prec, pos );
2020-06-03 04:33:18 -07:00
}
#endif
#define BUFFER_MINIMAL_GAP_SIZE (16u)
static void pcan_rx_can_frame( PCAN_RECORD_BUFFER_EX *prec, const can_message_t *msg )
{
2020-06-03 13:58:38 -07:00
uint8_t *ptr, pos = 0;
ptr = pcan_record_buffer_request( prec, BUFFER_MINIMAL_GAP_SIZE );
/* wow ! too fast too furious !!! */
if( !ptr )
{
if( !pcan_record_buffer_flush( prec ) )
{
pcan_device.can.err |= PCAN_USB_ERROR_RXQOVR;
return;
}
ptr = pcan_record_buffer_request( prec, BUFFER_MINIMAL_GAP_SIZE );
/* o_0 */
if( !ptr )
return;
}
/* recheck dlc for abnormal frames... */
uint8_t sl = msg->dlc > 8 ? 8: msg->dlc;
if( msg->flags & CAN_FLAG_EXTID )
{
sl |= PCAN_USB_STATUSLEN_EXT_ID;
}
if( msg->flags & CAN_FLAG_RTR )
{
sl |= PCAN_USB_STATUSLEN_RTR;
}
pack_u8( ptr, sl );
ptr += 1;
pos += 1;
if( sl & PCAN_USB_STATUSLEN_EXT_ID )
{
uint32_t id = ( msg->id & 0x1FFFFFFF ) << 3;
if( msg->flags & CAN_FLAG_ECHO )
{
id |= 0x01;
}
pack_u32( ptr, id );
ptr += 4;
pos += 4;
}
else
{
uint16_t id = ( msg->id & 0x7FF ) << 5;
if( msg->flags & CAN_FLAG_ECHO )
{
id |= 0x01;
}
pack_u16( ptr, id );
ptr += 2;
pos += 2;
}
/* timestamp to ticks, 1 tick is 42.666us */
uint16_t timestamp16 = msg->timestamp;
if( prec->ts_counter == 0 )
{
pack_u16( ptr, timestamp16 );
ptr += 2;
pos += 2;
}
else
{
pack_u8( ptr, timestamp16&0xFF );
ptr += 1;
pos += 1;
}
++prec->ts_counter;
if( !( sl & PCAN_USB_STATUSLEN_RTR ) )
{
memcpy( ptr, msg->data, msg->dlc );
ptr += msg->dlc;
pos += msg->dlc;
}
if( msg->flags & CAN_FLAG_ECHO )
{
pack_u8( ptr, msg->dummy );
ptr += 1;
pos += 1;
}
/* add new record */
pcan_record_buffer_commit( prec, pos );
2020-06-03 04:33:18 -07:00
}
static void pcan_rx_message( can_message_t *msg )
{
2020-06-03 13:58:38 -07:00
if( !pcan_device.bus_active )
return;
if( ( msg->flags & CAN_FLAG_ECHO ) == 0 )
{
pcan_led_set_mode( LED_CH0_RX, LED_MODE_BLINK_FAST, 237 );
}
2020-06-03 04:33:18 -07:00
pcan_rx_can_frame( &pcan_records, msg );
}
static void pcan_can_error( uint8_t err, uint8_t rx_err, uint8_t tx_err )
{
2020-06-03 13:58:38 -07:00
pcan_device.can.rx_err = rx_err;
pcan_device.can.tx_err = tx_err;
2020-06-03 04:33:18 -07:00
2020-06-03 13:58:38 -07:00
if( err & CAN_ERROR_FLAG_BUSOFF )
{
pcan_device.can.err |= PCAN_USB_ERROR_BUS_OFF;
}
2020-06-03 04:33:18 -07:00
}
static uint8_t pcan_decode_data_frame( uint8_t *ptr, uint16_t size, uint8_t flags )
{
2020-06-03 13:58:38 -07:00
can_message_t msg = { 0 };
uint8_t rec_len = flags & PCAN_USB_STATUSLEN_DLC;
/* return back frame to PC */
2020-06-03 04:33:18 -07:00
uint8_t srr_flag = 0;
2020-06-03 13:58:38 -07:00
if( !pcan_device.bus_active )
return 0;
2020-06-03 04:33:18 -07:00
if( flags & PCAN_USB_STATUSLEN_EXT_ID )
{
2020-06-03 13:58:38 -07:00
uint32_t id;
2020-06-03 04:33:18 -07:00
if( size < sizeof( uint32_t ) )
return 0;
id = unpack_u32( ptr );
srr_flag = id & 0x01;
id >>= 3;
2020-06-03 13:58:38 -07:00
msg.flags = CAN_FLAG_EXTID;
msg.id = id;
2020-06-03 04:33:18 -07:00
ptr += sizeof( uint32_t );
size -= sizeof( uint32_t );
}
else
{
2020-06-03 13:58:38 -07:00
uint16_t id;
2020-06-03 04:33:18 -07:00
if( size < sizeof( uint16_t ) )
return 0;
id = unpack_u16( ptr );
srr_flag = id & 0x01;
id >>= 5;
2020-06-03 13:58:38 -07:00
msg.id = id;
2020-06-03 04:33:18 -07:00
ptr += sizeof( uint16_t );
size -= sizeof( uint16_t );
}
/* check for MAX DLC */
msg.dlc = rec_len < 8 ? rec_len: 8;
if( size < rec_len )
return 0;
if( !( flags & PCAN_USB_STATUSLEN_RTR ) )
{
memcpy( msg.data, ptr, msg.dlc );
2020-06-03 13:58:38 -07:00
ptr += rec_len;
size -= rec_len;
}
else
{
msg.flags |= CAN_FLAG_RTR;
2020-06-03 04:33:18 -07:00
}
/* self receive flag ? */
if( srr_flag )
{
if( size < sizeof( uint8_t ) )
return 0;
2020-06-03 13:58:38 -07:00
msg.dummy = unpack_u8( ptr );
2020-06-03 04:33:18 -07:00
ptr += sizeof( uint8_t );
size -= sizeof( uint8_t );
}
2020-06-03 13:58:38 -07:00
msg.timestamp = pcan_timestamp_ticks();
2020-06-03 04:33:18 -07:00
2020-06-03 13:58:38 -07:00
pcan_led_set_mode( LED_CH0_TX, LED_MODE_BLINK_FAST, 237 );
2020-06-03 04:33:18 -07:00
if( pcan_can_send_message( &msg ) < 0 )
2020-06-03 13:58:38 -07:00
{
/* tx queue overflow ? */
pcan_device.can.err |= PCAN_USB_ERROR_TXFULL;
return size;
}
/* return back to PC */
if( srr_flag )
{
msg.flags |= CAN_FLAG_ECHO;
pcan_rx_message( &msg );
}
return size;
2020-06-03 04:33:18 -07:00
}
static void pcan_set_bitrate( uint8_t *ptr )
{
/* decode BTR1 */
volatile uint8_t tseg1 = (ptr[0] & 0xF) + 1;
volatile uint8_t tseg2 = ((ptr[0]>>4) & 0x7) + 1;
/* decode BTR0 */
volatile uint16_t brp = (ptr[1] & 0x3f) + 1;
2020-06-03 13:58:38 -07:00
volatile uint8_t sjw = ((ptr[1]>>6) & 0x3) + 1;
2020-06-03 04:33:18 -07:00
uint32_t bitrate = (((PCAN_USB_CRYSTAL_HZ/2)/brp)/(1/*tq*/ + tseg1 + tseg2 ));
2020-06-03 13:58:38 -07:00
(void)bitrate;
pcan_can_set_bitrate( brp, tseg1, tseg2, sjw );
2020-06-03 04:33:18 -07:00
}
/* create */
void pcan_timesync_event( PCAN_RECORD_BUFFER_EX *prec )
{
2020-06-03 13:58:38 -07:00
uint8_t sl = PCAN_USB_STATUSLEN_INTERNAL | 2 /* record data size */;
uint8_t *ptr, pos = 0;
2020-06-03 04:33:18 -07:00
2020-06-03 13:58:38 -07:00
ptr = pcan_record_buffer_request( prec, 5 );
if( !ptr )
return;
uint16_t timestamp16 = pcan_timestamp_ticks();
2020-06-03 04:33:18 -07:00
2020-06-03 13:58:38 -07:00
pack_u8( ptr, sl );
ptr += 1;
pos += 1;
2020-06-03 04:33:18 -07:00
2020-06-03 13:58:38 -07:00
/* function */
pack_u8( ptr, PCAN_USB_REC_TS );
ptr += 1;
pos += 1;
2020-06-03 04:33:18 -07:00
2020-06-03 13:58:38 -07:00
/* number */
pack_u8( ptr, 1 );
ptr += 1;
pos += 1;
2020-06-03 04:33:18 -07:00
2020-06-03 13:58:38 -07:00
/* timestamp */
pack_u16( ptr, timestamp16 );
ptr += 2;
pos += 2;
2020-06-03 04:33:18 -07:00
2020-06-03 13:58:38 -07:00
/* add new record */
pcan_record_buffer_commit( prec, pos );
2020-06-03 04:33:18 -07:00
2020-06-03 13:58:38 -07:00
/* error record */
pos = 0;
ptr = pcan_record_buffer_request( prec, 3 );
if( !ptr )
return;
2020-06-03 04:33:18 -07:00
2020-06-03 13:58:38 -07:00
pcan_device.can.err &= pcan_device.can.err_mask;
2020-06-03 04:33:18 -07:00
2020-06-03 13:58:38 -07:00
pack_u8( ptr, PCAN_USB_STATUSLEN_INTERNAL );
ptr += 1;
pos += 1;
2020-06-03 04:33:18 -07:00
2020-06-03 13:58:38 -07:00
pack_u8( ptr, PCAN_USB_REC_ERROR ); /* f */
ptr += 1;
pos += 1;
2020-06-03 04:33:18 -07:00
2020-06-03 13:58:38 -07:00
pack_u8( ptr, pcan_device.can.err ); /* n */
ptr += 1;
pos += 1;
2020-06-03 04:33:18 -07:00
2020-06-03 13:58:38 -07:00
/* add new record */
pcan_record_buffer_commit( prec, pos );
2020-06-03 04:33:18 -07:00
2020-06-03 13:58:38 -07:00
/* clean errors */
pcan_device.can.err = 0x00;
2020-06-03 04:33:18 -07:00
}
void pcan_protocol_process_command( uint8_t *ptr, uint16_t size )
{
2020-06-03 13:58:38 -07:00
PCAN_USB_PARAM *cmd = (void*)ptr;
if( size < PCAN_USB_MSG_HEADER_LEN )
return;
switch( cmd->number )
{
case PCAN_USB_EX0:
switch( cmd->function )
{
/* set mass storage mode */
case 0xC8:
break;
}
break;
case PCAN_USB_EX3:
switch( cmd->function )
{
/* CAN silient control*/
case 0x03:
pcan_device.can.silient = cmd->param[0];
pcan_can_set_silent( pcan_device.can.silient );
break;
}
break;
case PCAN_USB_SET:
switch( cmd->function )
{
/* BTR0 BTR1 */
case 0x01:
pcan_device.can.btr0 = cmd->param[1];
pcan_device.can.btr1 = cmd->param[0];
2020-06-03 04:33:18 -07:00
pcan_set_bitrate( cmd->param );
2020-06-03 13:58:38 -07:00
break;
/* set CAN on/off*/
case 0x03:
pcan_device.bus_active = cmd->param[0];
pcan_led_set_mode( LED_STAT, pcan_device.bus_active ? LED_MODE_BLINK_SLOW:LED_MODE_OFF, 0 );
pcan_can_set_bus_active( pcan_device.bus_active );
if( pcan_device.bus_active )
{
/* provide first timesync event */
pcan_device.last_timestamp_sync = pcan_timestamp_ticks();
pcan_timesync_event( &pcan_records );
}
/* led state */
pcan_led_set_mode( LED_CH0_TX, pcan_device.bus_active ? LED_MODE_ON:LED_MODE_OFF, 0 );
pcan_led_set_mode( LED_CH0_RX, pcan_device.bus_active ? LED_MODE_ON:LED_MODE_OFF, 0 );
break;
/* set device id 0-255 */
case 0x04:
pcan_device.device_id = cmd->param[0];
break;
/* unknown windows driver call */
case 0x05:
break;
/* sja1000 mode */
case 0x09:
pcan_device.can.sja1000_mode = cmd->param[0];
break;
/* ext VCC on/off */
case 0x0A:
pcan_device.ext_vcc_state = cmd->param[0];
break;
/* error frame mask */
case 0x0B:
pcan_device.can.err_mask = cmd->param[0];
break;
/* set led mode */
case 0x0C:
pcan_device.led_mode = cmd->param[0];
break;
}
break;
case PCAN_USB_GET:
switch( cmd->function )
{
/* BTR0 BTR1 */
case 0x01:
cmd->param[1] = pcan_device.can.btr0;
cmd->param[0] = pcan_device.can.btr1;
pcan_usb_send_command_buffer( cmd, sizeof( PCAN_USB_PARAM ) );
break;
/* quarts freq in Mhz */
case 0x02:
{
cmd->param[0] = pcan_device.can.quartz_freq/1000000u;
pcan_usb_send_command_buffer( cmd, sizeof( PCAN_USB_PARAM ) );
}
break;
/* get device id */
case 0x04:
{
/* default: 255 */
cmd->param[0] = pcan_device.device_id;
pcan_usb_send_command_buffer( cmd, sizeof( PCAN_USB_PARAM ) );
}
break;
/* serial */
case 0x06:
{
/* default: 4 bytes FFFFFFFF */
cmd->param[0] = (pcan_device.device_serial>>0x18)&0xFF;
cmd->param[1] = (pcan_device.device_serial>>0x10)&0xFF;
cmd->param[2] = (pcan_device.device_serial>>0x08)&0xFF;
cmd->param[3] = (pcan_device.device_serial>>0x00)&0xFF;
pcan_usb_send_command_buffer( cmd, sizeof( PCAN_USB_PARAM ) );
}
break;
/* get led */
case 0x0C:
cmd->param[0] = pcan_device.led_mode;
pcan_usb_send_command_buffer( cmd, sizeof( PCAN_USB_PARAM ) );
break;
}
break;
default:
assert( 0 );
break;
}
2020-06-03 04:33:18 -07:00
}
void pcan_protocol_process_data( uint8_t *ptr, uint16_t size )
{
2020-06-03 13:58:38 -07:00
uint8_t msg_prefix, msg_rec_count;
2020-06-03 04:33:18 -07:00
2020-06-03 13:58:38 -07:00
/* invalid frame header len ? */
if( size < PCAN_USB_MSG_HEADER_LEN )
return;
2020-06-03 04:33:18 -07:00
2020-06-03 13:58:38 -07:00
msg_prefix = *ptr++;
msg_rec_count = *ptr++;
2020-06-03 04:33:18 -07:00
2020-06-03 13:58:38 -07:00
(void)msg_prefix; /* avoid warning */
2020-06-03 04:33:18 -07:00
2020-06-03 13:58:38 -07:00
size -= PCAN_USB_MSG_HEADER_LEN;
2020-06-03 04:33:18 -07:00
2020-06-03 13:58:38 -07:00
for( uint8_t i = 0; i < msg_rec_count; i++ )
{
2020-06-03 04:33:18 -07:00
uint16_t eated = 0, sl;
2020-06-03 13:58:38 -07:00
if( size < 1 )
2020-06-03 04:33:18 -07:00
return;
sl = unpack_u8( ptr );
ptr += sizeof( uint8_t );
size -= sizeof( uint8_t );;
2020-06-03 13:58:38 -07:00
/* status, error frame */
2020-06-03 04:33:18 -07:00
if( sl & PCAN_USB_STATUSLEN_INTERNAL )
{
/* ??? abnormal, device support only CAN data frame */
eated = 0;
}
/* regular CAN data frame */
else
{
eated = pcan_decode_data_frame( ptr, size, sl );
}
2020-06-03 13:58:38 -07:00
/* ? error indicator */
2020-06-03 04:33:18 -07:00
if( !eated )
return;
eated = size - eated;
if( size < eated )
return;
2020-06-03 13:58:38 -07:00
ptr += eated;
2020-06-03 04:33:18 -07:00
size -= eated;
2020-06-03 13:58:38 -07:00
}
2020-06-03 04:33:18 -07:00
}
void pcan_protocol_init( void )
{
2020-06-03 13:58:38 -07:00
pcan_record_buffer_init( &pcan_records );
2020-06-03 04:33:18 -07:00
2020-06-03 13:58:38 -07:00
pcan_can_init();
2020-06-03 04:33:18 -07:00
pcan_can_install_rx_callback( pcan_rx_message );
2020-06-03 13:58:38 -07:00
pcan_can_install_error_callback( pcan_can_error );
2020-06-03 04:33:18 -07:00
}
void pcan_protocol_poll( void )
{
2020-06-03 13:58:38 -07:00
uint16_t ts_ms = pcan_timestamp_ticks();
if( pcan_device.bus_active )
{
/* each ~1000 ms */
if( ( ((uint16_t)( ts_ms - pcan_device.last_timestamp_sync )) >= PCAN_TICKS_FROM_US( 1000000u ) ) )
{
pcan_device.last_timestamp_sync = ts_ms;
pcan_timesync_event( &pcan_records );
}
}
pcan_record_buffer_flush( &pcan_records );
pcan_can_poll();
2020-06-03 04:33:18 -07:00
}