integrator_serial_with_ints.c
来自「eCos操作系统源码」· C语言 代码 · 共 412 行 · 第 1/2 页
C
412 行
// ----------v----------v----------v----------v---------- // NOTE: MUST BE WRITTEN LAST (AFTER UARTLCR_M & UARTLCR_L) // ----------^----------^----------^----------^---------- _lcr = select_word_length[new_config->word_length - CYGNUM_SERIAL_WORD_LENGTH_5] | select_stop_bits[new_config->stop] | select_parity[new_config->parity] | AMBA_UARTLCR_H_FEN ; IO_WRITE(port + AMBA_UARTLCR_H, _lcr); // finally, enable the uart IO_WRITE(port + AMBA_UARTCR, (AMBA_UARTCR_RIE | AMBA_UARTCR_RTIE | AMBA_UARTCR_UARTEN)); // save the configuration if (new_config != &chan->config) { chan->config = *new_config; } // success return true;}// Function to initialize the device. Called at bootstrap time.static bool integrator_serial_init(struct cyg_devtab_entry *tab){ serial_channel *chan = (serial_channel *)tab->priv; integrator_serial_info *integrator_chan = (integrator_serial_info *)chan->dev_priv;#ifdef CYGDBG_IO_INIT diag_printf("INTEGRATOR SERIAL init - dev: %x.%d\n", integrator_chan->base, integrator_chan->int_num);#endif (chan->callbacks->serial_init)(chan); // Really only required for interrupt driven devices if (chan->out_cbuf.len != 0) { cyg_drv_interrupt_create(integrator_chan->int_num, 99, // Priority - what goes here? (cyg_addrword_t)chan, // Data item passed to interrupt handler integrator_serial_ISR, integrator_serial_DSR, &integrator_chan->serial_interrupt_handle, &integrator_chan->serial_interrupt); cyg_drv_interrupt_attach(integrator_chan->serial_interrupt_handle); cyg_drv_interrupt_unmask(integrator_chan->int_num); } integrator_serial_config_port(chan, &chan->config, true); return true;}// This routine is called when the device is "looked" up (i.e. attached)static Cyg_ErrNo integrator_serial_lookup(struct cyg_devtab_entry **tab, struct cyg_devtab_entry *sub_tab, const char *name){ serial_channel *chan = (serial_channel *)(*tab)->priv; (chan->callbacks->serial_init)(chan); // Really only required for interrupt driven devices return ENOERR;}// Send a character to the device output buffer.// Return 'true' if character is sent to devicestatic boolintegrator_serial_putc(serial_channel *chan, unsigned char c){ integrator_serial_info *integrator_chan = (integrator_serial_info *)chan->dev_priv; unsigned int status = GET_STATUS(integrator_chan->base) ; if (TX_READY(status)) {// Transmit buffer is empty PUT_CHAR(integrator_chan->base, c) ; chars_tx++ ; return true; } else {// No space return false; }}// Fetch a character from the device input buffer, waiting if necessarystatic unsigned char integrator_serial_getc(serial_channel *chan){ unsigned char c; integrator_serial_info *integrator_chan = (integrator_serial_info *)chan->dev_priv; unsigned int status ; do { status = GET_STATUS(integrator_chan->base) ; } while (!RX_DATA(status)) ; // Wait for char chars_rx++ ; // get it c = GET_CHAR(integrator_chan->base) ; return c;}// Set up the device characteristics; baud rate, etc.static Cyg_ErrNointegrator_serial_set_config(serial_channel *chan, cyg_uint32 key, const void *xbuf, cyg_uint32 *len){ integrator_serial_info *integrator_chan = (integrator_serial_info *)chan->dev_priv; switch (key) { case CYG_IO_SET_CONFIG_SERIAL_INFO: { cyg_serial_info_t *config = (cyg_serial_info_t *)xbuf; if ( *len < sizeof(cyg_serial_info_t) ) { return -EINVAL; } *len = sizeof(cyg_serial_info_t); if ( true != integrator_serial_config_port(chan, config, false) ) return -EINVAL; } break;#ifdef CYGOPT_IO_SERIAL_FLOW_CONTROL_HW#ifdef FIXME case CYG_IO_SET_CONFIG_SERIAL_HW_RX_FLOW_THROTTLE: { volatile struct serial_port *port = (volatile struct serial_port *)integrator_chan->base; cyg_uint32 *f = (cyg_uint32 *)xbuf; unsigned char mask=0; if ( *len < *f ) return -EINVAL; if ( chan->config.flags & CYGNUM_SERIAL_FLOW_RTSCTS_RX ) mask = MCR_RTS; if ( chan->config.flags & CYGNUM_SERIAL_FLOW_DSRDTR_RX ) mask |= MCR_DTR; if (*f) // we should throttle port->REG_mcr &= ~mask; else // we should no longer throttle port->REG_mcr |= mask; } break; case CYG_IO_SET_CONFIG_SERIAL_HW_FLOW_CONFIG: // Nothing to do because we do support both RTSCTS and DSRDTR flow // control. // Other targets would clear any unsupported flags here. // We just return ENOERR. break;#else#error "Flow control for Integrator not integrated!"#endif#endif default: return -EINVAL; } return ENOERR;}// Enable the transmitter on the devicestatic voidintegrator_serial_start_xmit(serial_channel *chan){ integrator_serial_info *integrator_chan = (integrator_serial_info *)chan->dev_priv; IO_WRITE(integrator_chan->base + AMBA_UARTCR, IO_READ(integrator_chan->base + AMBA_UARTCR) | AMBA_UARTCR_TIE);}// Disable the transmitter on the devicestatic void integrator_serial_stop_xmit(serial_channel *chan){ integrator_serial_info *integrator_chan = (integrator_serial_info *)chan->dev_priv; IO_WRITE(integrator_chan->base + AMBA_UARTCR, IO_READ(integrator_chan->base + AMBA_UARTCR) & ~AMBA_UARTCR_TIE);}// Serial I/O - low level interrupt handler (ISR)static cyg_uint32 integrator_serial_ISR(cyg_vector_t vector, cyg_addrword_t data){ serial_channel *chan = (serial_channel *)data; integrator_serial_info *integrator_chan = (integrator_serial_info *)chan->dev_priv; cyg_drv_interrupt_mask(integrator_chan->int_num); cyg_drv_interrupt_acknowledge(integrator_chan->int_num); return CYG_ISR_CALL_DSR; // Cause DSR to be run}// Serial I/O - high level interrupt handler (DSR)static void integrator_serial_DSR(cyg_vector_t vector, cyg_ucount32 count, cyg_addrword_t data){ serial_channel *chan = (serial_channel *)data; integrator_serial_info *integrator_chan = (integrator_serial_info *)chan->dev_priv; volatile unsigned char isr = GET_INTERRUPT_STATUS(integrator_chan->base) ; while ((isr & (AMBA_UARTIIR_RTIS | AMBA_UARTIIR_TIS | AMBA_UARTIIR_RIS)) != 0) { if (isr & AMBA_UARTIIR_TIS) { (chan->callbacks->xmt_char)(chan); } else if (isr & AMBA_UARTIIR_RTIS) { chars_rx++ ; (chan->callbacks->rcv_char)(chan, GET_CHAR(integrator_chan->base)); } else if (isr & AMBA_UARTIIR_RIS) { chars_rx++ ; (chan->callbacks->rcv_char)(chan, GET_CHAR(integrator_chan->base)); } isr = GET_INTERRUPT_STATUS(integrator_chan->base) ; } cyg_drv_interrupt_unmask(integrator_chan->int_num);}#endif
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?