/* * This file is part of the libopencm3 project. * * Copyright (C) 2010 Thomas Otto * Copyright (C) 2010 Piotr Esden-Tempski * * This library is free software: you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public License * along with this library. If not, see . */ #include #include #include #include #include #include struct can_tx_msg { u32 std_id; u32 ext_id; u8 ide; u8 rtr; u8 dlc; u8 data[8]; }; struct can_rx_msg { u32 std_id; u32 ext_id; u8 ide; u8 rtr; u8 dlc; u8 data[8]; u8 fmi; }; struct can_tx_msg can_tx_msg; struct can_rx_msg can_rx_msg; void gpio_setup(void) { /* Enable GPIOA clock. */ rcc_peripheral_enable_clock(&RCC_APB2ENR, RCC_APB2ENR_IOPAEN); /* Enable GPIOB clock. */ rcc_peripheral_enable_clock(&RCC_APB2ENR, RCC_APB2ENR_IOPBEN); gpio_set(GPIOA, GPIO6); /* LED0 off */ gpio_set(GPIOA, GPIO7); /* LED1 off */ gpio_set(GPIOB, GPIO0); /* LED2 off */ gpio_set(GPIOB, GPIO1); /* LED3 off */ /* Set GPIO6/7 (in GPIO port A) to 'output push-pull' for the LEDs. */ gpio_set_mode(GPIOA, GPIO_MODE_OUTPUT_50_MHZ, GPIO_CNF_OUTPUT_PUSHPULL, GPIO6); gpio_set_mode(GPIOA, GPIO_MODE_OUTPUT_50_MHZ, GPIO_CNF_OUTPUT_PUSHPULL, GPIO7); /* Set GPIO0/1 (in GPIO port B) to 'output push-pull' for the LEDs. */ gpio_set_mode(GPIOB, GPIO_MODE_OUTPUT_50_MHZ, GPIO_CNF_OUTPUT_PUSHPULL, GPIO0); gpio_set_mode(GPIOB, GPIO_MODE_OUTPUT_50_MHZ, GPIO_CNF_OUTPUT_PUSHPULL, GPIO1); } void systick_setup(void) { /* 72MHz / 8 => 9000000 counts per second */ systick_set_clocksource(STK_CTRL_CLKSOURCE_AHB_DIV8); /* 9000000/9000 = 1000 overflows per second - every 1ms one interrupt */ systick_set_reload(9000); systick_interrupt_enable(); /* Start counting. */ systick_counter_enable(); } void can_setup(void) { /* Enable peripheral clocks. */ rcc_peripheral_enable_clock(&RCC_APB2ENR, RCC_APB2ENR_AFIOEN); rcc_peripheral_enable_clock(&RCC_APB2ENR, RCC_APB2ENR_IOPAEN); rcc_peripheral_enable_clock(&RCC_APB1ENR, RCC_APB1ENR_CANEN); /* Configure CAN pin: RX (input pull-up). */ gpio_set_mode(GPIOA, GPIO_MODE_INPUT, GPIO_CNF_INPUT_PULL_UPDOWN, GPIO_CAN_RX); gpio_set(GPIOA, GPIO_CAN_RX); /* Configure CAN pin: TX. */ gpio_set_mode(GPIOA, GPIO_MODE_OUTPUT_50_MHZ, GPIO_CNF_OUTPUT_ALTFN_PUSHPULL, GPIO_CAN_TX); /* NVIC setup. */ nvic_enable_irq(NVIC_USB_LP_CAN_RX0_IRQ); nvic_set_priority(NVIC_USB_LP_CAN_RX0_IRQ, 1); /* Reset CAN. */ can_reset(CAN1); /* CAN cell init. */ if (can_init(CAN1, false, /* TTCM: Time triggered comm mode? */ true, /* ABOM: Automatic bus-off management? */ false, /* AWUM: Automatic wakeup mode? */ false, /* NART: No automatic retransmission? */ false, /* RFLM: Receive FIFO locked mode? */ false, /* TXFP: Transmit FIFO priority? */ CAN_BTR_SJW_1TQ, CAN_BTR_TS1_3TQ, CAN_BTR_TS2_4TQ, 12)) /* BRP+1: Baud rate prescaler */ { gpio_set(GPIOA, GPIO6); /* LED0 off */ gpio_set(GPIOA, GPIO7); /* LED1 off */ gpio_set(GPIOB, GPIO0); /* LED2 off */ gpio_clear(GPIOB, GPIO1); /* LED3 on */ /* Die because we failed to initialize. */ while (1) __asm__("nop"); } /* CAN filter 0 init. */ can_filter_id_mask_32bit_init(CAN1, 0, /* Filter ID */ 0, /* CAN ID */ 0, /* CAN ID mask */ 0, /* FIFO assignment (here: FIFO0) */ true); /* Enable the filter. */ /* Enable CAN RX interrupt. */ can_enable_irq(CAN1, CAN_IER_FMPIE0); } void sys_tick_handler(void) { static int temp32 = 0; static u8 data[8] = {0, 1, 2, 0, 0, 0, 0, 0}; /* We call this handler every 1ms so 1000ms = 1s on/off. */ if (++temp32 != 1000) return; temp32 = 0; /* Transmit CAN frame. */ data[0]++; if (can_transmit(CAN1, 0, /* (EX/ST)ID: CAN ID */ false, /* IDE: CAN ID extended? */ false, /* RTR: Request transmit? */ 8, /* DLC: Data length */ data) == -1) { gpio_set(GPIOA, GPIO6); /* LED0 off */ gpio_set(GPIOA, GPIO7); /* LED1 off */ gpio_clear(GPIOB, GPIO0); /* LED2 on */ gpio_set(GPIOB, GPIO1); /* LED3 off */ } } void usb_lp_can_rx0_isr(void) { u32 id, fmi; bool ext, rtr; u8 length, data[8]; can_receive(CAN1, 0, false, &id, &ext, &rtr, &fmi, &length, data); if (data[0] & 1) gpio_clear(GPIOA, GPIO6); else gpio_set(GPIOA, GPIO6); if (data[0] & 2) gpio_clear(GPIOA, GPIO7); else gpio_set(GPIOA, GPIO7); if (data[0] & 4) gpio_clear(GPIOB, GPIO0); else gpio_set(GPIOB, GPIO0); if (data[0] & 8) gpio_clear(GPIOB, GPIO1); else gpio_set(GPIOB, GPIO1); can_fifo_release(CAN1, 0); } int main(void) { rcc_clock_setup_in_hse_8mhz_out_72mhz(); gpio_setup(); can_setup(); systick_setup(); while (1); /* Halt. */ return 0; }