Hi, I make PATCH for the 16550A serial drivers. It allow to assert RTS signal during transmit in EIA-485 half duplex mode for 16550A compatible controllers.
I have also modified the libmodbus-2.0.3 to be used with xenomai if anyone is intesested// in this lib, please let me know Thanks, Alexandre -- Alexandre COFFIGNAL, Chef de Projet Email: alexandre.coffig...@cenosys.com <mailto:alexandre.coffig...@cenosys.com> ------------------------------------ CénoSYS <http://www.cenosys.com> 10, Rue Xavier Bichat F-72000 Le MANS Tel: +33(0) 243 511 797 web : http://www.cenosys.com ------------------------------------ CONFIDENTIALITE : Ce message électronique et tous les fichiers attachés qu'il contient sont confidentiels et destinés exclusivement à l'usage de la personne à laquelle ils sont adressés. Si vous avez reçu ce message par erreur, merci de le retourner immédiatement a son émetteur sans en conserver de copie. CONFIDENTIALITY : This e-mail and any attachments are confidential and may also be privileged. If you are not the named recipient, please notify the sender immediately and do not disclose the contents to an other person, use it for any purpose, or store or copy the information in any medium.
>From 0aa88d5d4fe5ab43f3069ab4dc8125d4a866f24c Mon Sep 17 00:00:00 2001 From: Alexandre Coffignal <alexandre.coffig...@cenosys.com> Date: Wed, 20 Jan 2010 15:13:39 +0100 Subject: [PATCH] rtdm : Allow to assert RTS signal during transmit in EIA-485 half duplex mode for 16550A compatible controllers. --- include/rtdm/rtserial.h | 34 ++++++++++++ ksrc/drivers/serial/16550A.c | 115 ++++++++++++++++++++++++++++++++++++++++- ksrc/drivers/serial/Kconfig | 7 +++ 3 files changed, 153 insertions(+), 3 deletions(-) diff --git a/include/rtdm/rtserial.h b/include/rtdm/rtserial.h index 48712b2..3a7b091 100644 --- a/include/rtdm/rtserial.h +++ b/include/rtdm/rtserial.h @@ -127,8 +127,13 @@ #define RTSER_NO_HAND 0x00 #define RTSER_RTSCTS_HAND 0x01 #define RTSER_DEF_HAND RTSER_NO_HAND +#ifdef CONFIG_XENO_DRIVERS_16550A_RS485_HALF +#define RTSER_RTS_RS485_HALF 0x02 +#endif /** @} */ + + /*! * @anchor RTSER_FIFO_xxx @name RTSER_FIFO_xxx * Reception FIFO interrupt threshold @@ -503,4 +508,33 @@ typedef struct rtser_event { /** @} */ +#ifdef CONFIG_XENO_DRIVERS_16550A_RS485_HALF + +typedef struct rs485_config +{ + nanosecs_rel_t delay_after_send; +}rs485_config_t; + +#define RS485_DEF_DELAY_AFTER_SEND 100000 //100uS + +/** + * Set config for rs485 half duplex mode + * + * @param[in] arg New rs485 config content + * + * @return 0 on success, otherwise negative error code + * + * Environments: + * + * This service can be called from: + * + * - Kernel module initialization/cleanup code + * - Kernel-based task + * - User-space task (RT, non-RT) + * + * Rescheduling: never. + */ +#define RTSER_RTIOC_RS485_SET_CONFIG \ + _IOW(RTIOC_TYPE_SERIAL, 0x07, struct rs485_config) +#endif #endif /* _RTSERIAL_H */ diff --git a/ksrc/drivers/serial/16550A.c b/ksrc/drivers/serial/16550A.c index db8a515..45c5cc3 100644 --- a/ksrc/drivers/serial/16550A.c +++ b/ksrc/drivers/serial/16550A.c @@ -107,6 +107,12 @@ struct rt_16550_context { int mcr_status; /* MCR cache */ int status; /* cache for LSR + soft-states */ int saved_errors; /* error cache for RTIOC_GET_STATUS */ +#ifdef CONFIG_XENO_DRIVERS_16550A_RS485_HALF + size_t bytes_send; /* bytes sent on last tx it*/ + rtdm_task_t rts_task; /* task used to assert RTS signal during transmit(rs485)*/ + rtdm_sem_t itTx; /* raised to unblock rts task (rs485) */ + rs485_config_t rs485_config; /* delay after TX frame before rts down in nanosecond(rs485) */ +#endif }; static const struct rtser_config default_config = { @@ -204,8 +210,16 @@ static inline void rt_16550_tx_interrupt(struct rt_16550_context *ctx) ctx->out_head &= (OUT_BUFFER_SIZE - 1); } } -} +#ifdef CONFIG_XENO_DRIVERS_16550A_RS485_HALF + if(ctx->config.handshake==RTSER_RTS_RS485_HALF) + { + /* up to unblock rts task in rs485 half duplex mode */ + ctx->bytes_send=ctx->tx_fifo-count; + rtdm_sem_up(&ctx->itTx); + } +#endif +} static inline void rt_16550_stat_interrupt(struct rt_16550_context *ctx) { unsigned long base = ctx->base_addr; @@ -409,10 +423,15 @@ static int rt_16550_set_config(struct rt_16550_context *ctx, if (testbits(config->config_mask, RTSER_SET_HANDSHAKE)) { /* change handshake atomically */ rtdm_lock_get_irqsave(&ctx->lock, lock_ctx); - ctx->config.handshake = config->handshake; switch (ctx->config.handshake) { +#ifdef CONFIG_XENO_DRIVERS_16550A_RS485_HALF + case RTSER_RTS_RS485_HALF: + ctx->mcr_status = + RTSER_MCR_DTR | RTSER_MCR_RTS | RTSER_MCR_OUT2; + break; +#endif case RTSER_RTSCTS_HAND: // ...? @@ -437,6 +456,51 @@ void rt_16550_cleanup_ctx(struct rt_16550_context *ctx) rtdm_mutex_destroy(&ctx->out_lock); } +#ifdef CONFIG_XENO_DRIVERS_16550A_RS485_HALF +#define RTS_STD_PRIO 1 + +/* RS485 + * allow to put in transmit mode explicitly by asserting RTC signal to the driver during + * transmission + * + * This allows EIA-485 to implement linear topologies using only two wires + * + */ +void rts_task(void *arg) +{ + struct rt_16550_context *ctx= (struct rt_16550_context *)arg; + int ret; + nanosecs_rel_t delay ; + char event=1; + + while(1){ + //wait sem up from itTx + ret =rtdm_sem_down (&ctx->itTx); + if(ret) + printk("error rts_task... sem down error %d\n",ret); + + do{ + //wait end of transmission nbit + * fifo size / baud rate + delay=11*ctx->bytes_send*1000000/ctx->config.baud_rate; + ret=rtdm_sem_timeddown(&ctx->itTx, delay*1000+ctx->rs485_config.delay_after_send,NULL); + if(ret){ + if(ret!=-ETIMEDOUT){ + printk("error rts_task rtdm_event_timedwait... error %d\n",ret); + } + //no new rtdm event itTx receive during the transfert + event=0; + } + else//new rtdm event itTx receive during the transfert, still waiting + event=1; + + }while(event); + //down RTS + ctx->mcr_status &= ~RTSER_MCR_RTS; + rt_16550_reg_out(rt_16550_io_mode_from_ctx(ctx),ctx->base_addr, MCR, ctx->mcr_status); + } +} +#endif + int rt_16550_open(struct rtdm_dev_context *context, rtdm_user_info_t * user_info, int oflags) { @@ -458,7 +522,9 @@ int rt_16550_open(struct rtdm_dev_context *context, rt_16550_init_io_ctx(dev_id, ctx); ctx->tx_fifo = tx_fifo[dev_id]; - +#ifdef CONFIG_XENO_DRIVERS_16550A_RS485_HALF + ctx->rs485_config.delay_after_send=RS485_DEF_DELAY_AFTER_SEND; +#endif ctx->in_head = 0; ctx->in_tail = 0; ctx->in_npend = 0; @@ -475,6 +541,7 @@ int rt_16550_open(struct rtdm_dev_context *context, ctx->status = 0; ctx->saved_errors = 0; + rt_16550_set_config(ctx, &default_config, &dummy); err = rtdm_irq_request(&ctx->irq_handle, irq[dev_id], @@ -493,6 +560,17 @@ int rt_16550_open(struct rtdm_dev_context *context, rtdm_lock_get_irqsave(&ctx->lock, lock_ctx); +#ifdef CONFIG_XENO_DRIVERS_16550A_RS485_HALF + /* create semaphore used to unblock rts_task (rs485)*/ + rtdm_sem_init(&ctx->itTx, 0); + /* create task used to assert RTS signal during transmit in (rs485)*/ + err = rtdm_task_init(&ctx->rts_task, "rts_task", &rts_task, ctx, RTS_STD_PRIO, 0); + if (err) { + printk("error rtdm_task_init\n"); + return 0; + } +#endif + /* enable interrupts */ ctx->ier_status = IER_RX; rt_16550_reg_out(rt_16550_io_mode_from_ctx(ctx), ctx->base_addr, IER, @@ -535,6 +613,11 @@ int rt_16550_close(struct rtdm_dev_context *context, rtdm_irq_free(&ctx->irq_handle); +#ifdef CONFIG_XENO_DRIVERS_16550A_RS485_HALF + rtdm_sem_destroy(&ctx->itTx); + rtdm_task_destroy(&ctx->rts_task); +#endif + rt_16550_cleanup_ctx(ctx); if (in_history) { @@ -802,6 +885,24 @@ int rt_16550_ioctl(struct rtdm_dev_context *context, rtdm_lock_put_irqrestore(&ctx->lock, lock_ctx); break; } +#ifdef CONFIG_XENO_DRIVERS_16550A_RS485_HALF + case RTSER_RTIOC_RS485_SET_CONFIG: { + struct rs485_config *config; + struct rs485_config config_buf; + config = (struct rtser_config *)arg; + if (user_info) { + err = rtdm_safe_copy_from_user(user_info, &config_buf, + arg, + sizeof(struct rs485_config)); + if (err) + return err; + + config = &config_buf; + ctx->rs485_config.delay_after_send=config->delay_after_send; + } + break; +#endif + } default: err = -ENOTTY; @@ -1005,6 +1106,14 @@ ssize_t rt_16550_write(struct rtdm_dev_context * context, if (ret) return ret; +#ifdef CONFIG_XENO_DRIVERS_16550A_RS485_HALF + if(ctx->config.handshake==RTSER_RTS_RS485_HALF) + { + //assert RTS signal during transmit in rs485 half duplex mode + ctx->mcr_status |= RTSER_MCR_RTS; + rt_16550_reg_out(rt_16550_io_mode_from_ctx(ctx),ctx->base_addr, MCR, ctx->mcr_status); + } +#endif while (nbyte > 0) { rtdm_lock_get_irqsave(&ctx->lock, lock_ctx); diff --git a/ksrc/drivers/serial/Kconfig b/ksrc/drivers/serial/Kconfig index afbaabf..95337bf 100644 --- a/ksrc/drivers/serial/Kconfig +++ b/ksrc/drivers/serial/Kconfig @@ -39,4 +39,11 @@ config XENO_DRIVERS_16550A_ANY endchoice +config XENO_DRIVERS_16550A_RS485_HALF + depends on XENO_DRIVERS_16550A + bool "EIA-485 half duplex" + help + Assert RTS signal during transmit in EIA-485 half duplex mode + for 16550A compatible controllers. + endmenu -- 1.6.0.4
_______________________________________________ Xenomai-core mailing list Xenomai-core@gna.org https://mail.gna.org/listinfo/xenomai-core