|
@@ -1391,7 +1391,7 @@ static inline void e100_disable_rx_irq(struct e100_serial *info)
|
|
#if defined(CONFIG_ETRAX_RS485)
|
|
#if defined(CONFIG_ETRAX_RS485)
|
|
/* Enable RS-485 mode on selected port. This is UGLY. */
|
|
/* Enable RS-485 mode on selected port. This is UGLY. */
|
|
static int
|
|
static int
|
|
-e100_enable_rs485(struct tty_struct *tty,struct rs485_control *r)
|
|
|
|
|
|
+e100_enable_rs485(struct tty_struct *tty, struct serial_rs485 *r)
|
|
{
|
|
{
|
|
struct e100_serial * info = (struct e100_serial *)tty->driver_data;
|
|
struct e100_serial * info = (struct e100_serial *)tty->driver_data;
|
|
|
|
|
|
@@ -1409,13 +1409,11 @@ e100_enable_rs485(struct tty_struct *tty,struct rs485_control *r)
|
|
CONFIG_ETRAX_RS485_LTC1387_RXEN_PORT_G_BIT, 1);
|
|
CONFIG_ETRAX_RS485_LTC1387_RXEN_PORT_G_BIT, 1);
|
|
#endif
|
|
#endif
|
|
|
|
|
|
- info->rs485.rts_on_send = 0x01 & r->rts_on_send;
|
|
|
|
- info->rs485.rts_after_sent = 0x01 & r->rts_after_sent;
|
|
|
|
|
|
+ info->rs485.flags = r->flags;
|
|
if (r->delay_rts_before_send >= 1000)
|
|
if (r->delay_rts_before_send >= 1000)
|
|
info->rs485.delay_rts_before_send = 1000;
|
|
info->rs485.delay_rts_before_send = 1000;
|
|
else
|
|
else
|
|
info->rs485.delay_rts_before_send = r->delay_rts_before_send;
|
|
info->rs485.delay_rts_before_send = r->delay_rts_before_send;
|
|
- info->rs485.enabled = r->enabled;
|
|
|
|
/* printk("rts: on send = %i, after = %i, enabled = %i",
|
|
/* printk("rts: on send = %i, after = %i, enabled = %i",
|
|
info->rs485.rts_on_send,
|
|
info->rs485.rts_on_send,
|
|
info->rs485.rts_after_sent,
|
|
info->rs485.rts_after_sent,
|
|
@@ -1430,17 +1428,18 @@ e100_write_rs485(struct tty_struct *tty,
|
|
const unsigned char *buf, int count)
|
|
const unsigned char *buf, int count)
|
|
{
|
|
{
|
|
struct e100_serial * info = (struct e100_serial *)tty->driver_data;
|
|
struct e100_serial * info = (struct e100_serial *)tty->driver_data;
|
|
- int old_enabled = info->rs485.enabled;
|
|
|
|
|
|
+ int old_value = (info->rs485.flags) & SER_RS485_ENABLED;
|
|
|
|
|
|
/* rs485 is always implicitly enabled if we're using the ioctl()
|
|
/* rs485 is always implicitly enabled if we're using the ioctl()
|
|
- * but it doesn't have to be set in the rs485_control
|
|
|
|
|
|
+ * but it doesn't have to be set in the serial_rs485
|
|
* (to be backward compatible with old apps)
|
|
* (to be backward compatible with old apps)
|
|
* So we store, set and restore it.
|
|
* So we store, set and restore it.
|
|
*/
|
|
*/
|
|
- info->rs485.enabled = 1;
|
|
|
|
|
|
+ info->rs485.flags |= SER_RS485_ENABLED;
|
|
/* rs_write now deals with RS485 if enabled */
|
|
/* rs_write now deals with RS485 if enabled */
|
|
count = rs_write(tty, buf, count);
|
|
count = rs_write(tty, buf, count);
|
|
- info->rs485.enabled = old_enabled;
|
|
|
|
|
|
+ if (!old_value)
|
|
|
|
+ info->rs485.flags &= ~(SER_RS485_ENABLED);
|
|
return count;
|
|
return count;
|
|
}
|
|
}
|
|
|
|
|
|
@@ -1451,7 +1450,7 @@ static void rs485_toggle_rts_timer_function(unsigned long data)
|
|
struct e100_serial *info = (struct e100_serial *)data;
|
|
struct e100_serial *info = (struct e100_serial *)data;
|
|
|
|
|
|
fast_timers_rs485[info->line].function = NULL;
|
|
fast_timers_rs485[info->line].function = NULL;
|
|
- e100_rts(info, info->rs485.rts_after_sent);
|
|
|
|
|
|
+ e100_rts(info, (info->rs485.flags & SER_RS485_RTS_AFTER_SEND));
|
|
#if defined(CONFIG_ETRAX_RS485_DISABLE_RECEIVER)
|
|
#if defined(CONFIG_ETRAX_RS485_DISABLE_RECEIVER)
|
|
e100_enable_rx(info);
|
|
e100_enable_rx(info);
|
|
e100_enable_rx_irq(info);
|
|
e100_enable_rx_irq(info);
|
|
@@ -1647,7 +1646,7 @@ transmit_chars_dma(struct e100_serial *info)
|
|
info->tr_running = 0;
|
|
info->tr_running = 0;
|
|
|
|
|
|
#if defined(CONFIG_ETRAX_RS485) && defined(CONFIG_ETRAX_FAST_TIMER)
|
|
#if defined(CONFIG_ETRAX_RS485) && defined(CONFIG_ETRAX_FAST_TIMER)
|
|
- if (info->rs485.enabled) {
|
|
|
|
|
|
+ if (info->rs485.flags & SER_RS485_ENABLED) {
|
|
/* Set a short timer to toggle RTS */
|
|
/* Set a short timer to toggle RTS */
|
|
start_one_shot_timer(&fast_timers_rs485[info->line],
|
|
start_one_shot_timer(&fast_timers_rs485[info->line],
|
|
rs485_toggle_rts_timer_function,
|
|
rs485_toggle_rts_timer_function,
|
|
@@ -2577,7 +2576,7 @@ static void handle_ser_tx_interrupt(struct e100_serial *info)
|
|
info->icount.tx++;
|
|
info->icount.tx++;
|
|
if (info->xmit.head == info->xmit.tail) {
|
|
if (info->xmit.head == info->xmit.tail) {
|
|
#if defined(CONFIG_ETRAX_RS485) && defined(CONFIG_ETRAX_FAST_TIMER)
|
|
#if defined(CONFIG_ETRAX_RS485) && defined(CONFIG_ETRAX_FAST_TIMER)
|
|
- if (info->rs485.enabled) {
|
|
|
|
|
|
+ if (info->rs485.flags & SER_RS485_ENABLED) {
|
|
/* Set a short timer to toggle RTS */
|
|
/* Set a short timer to toggle RTS */
|
|
start_one_shot_timer(&fast_timers_rs485[info->line],
|
|
start_one_shot_timer(&fast_timers_rs485[info->line],
|
|
rs485_toggle_rts_timer_function,
|
|
rs485_toggle_rts_timer_function,
|
|
@@ -3218,7 +3217,7 @@ rs_write(struct tty_struct *tty,
|
|
#if defined(CONFIG_ETRAX_RS485)
|
|
#if defined(CONFIG_ETRAX_RS485)
|
|
struct e100_serial *info = (struct e100_serial *)tty->driver_data;
|
|
struct e100_serial *info = (struct e100_serial *)tty->driver_data;
|
|
|
|
|
|
- if (info->rs485.enabled)
|
|
|
|
|
|
+ if (info->rs485.flags & SER_RS485_ENABLED)
|
|
{
|
|
{
|
|
/* If we are in RS-485 mode, we need to toggle RTS and disable
|
|
/* If we are in RS-485 mode, we need to toggle RTS and disable
|
|
* the receiver before initiating a DMA transfer
|
|
* the receiver before initiating a DMA transfer
|
|
@@ -3228,7 +3227,7 @@ rs_write(struct tty_struct *tty,
|
|
fast_timers_rs485[info->line].function = NULL;
|
|
fast_timers_rs485[info->line].function = NULL;
|
|
del_fast_timer(&fast_timers_rs485[info->line]);
|
|
del_fast_timer(&fast_timers_rs485[info->line]);
|
|
#endif
|
|
#endif
|
|
- e100_rts(info, info->rs485.rts_on_send);
|
|
|
|
|
|
+ e100_rts(info, (info->rs485.flags & SER_RS485_RTS_ON_SEND));
|
|
#if defined(CONFIG_ETRAX_RS485_DISABLE_RECEIVER)
|
|
#if defined(CONFIG_ETRAX_RS485_DISABLE_RECEIVER)
|
|
e100_disable_rx(info);
|
|
e100_disable_rx(info);
|
|
e100_enable_rx_irq(info);
|
|
e100_enable_rx_irq(info);
|
|
@@ -3242,7 +3241,7 @@ rs_write(struct tty_struct *tty,
|
|
count = rs_raw_write(tty, buf, count);
|
|
count = rs_raw_write(tty, buf, count);
|
|
|
|
|
|
#if defined(CONFIG_ETRAX_RS485)
|
|
#if defined(CONFIG_ETRAX_RS485)
|
|
- if (info->rs485.enabled)
|
|
|
|
|
|
+ if (info->rs485.flags & SER_RS485_ENABLED)
|
|
{
|
|
{
|
|
unsigned int val;
|
|
unsigned int val;
|
|
/* If we are in RS-485 mode the following has to be done:
|
|
/* If we are in RS-485 mode the following has to be done:
|
|
@@ -3263,7 +3262,7 @@ rs_write(struct tty_struct *tty,
|
|
get_lsr_info(info, &val);
|
|
get_lsr_info(info, &val);
|
|
}while (!(val & TIOCSER_TEMT));
|
|
}while (!(val & TIOCSER_TEMT));
|
|
|
|
|
|
- e100_rts(info, info->rs485.rts_after_sent);
|
|
|
|
|
|
+ e100_rts(info, (info->rs485.flags & SER_RS485_RTS_AFTER_SEND));
|
|
|
|
|
|
#if defined(CONFIG_ETRAX_RS485_DISABLE_RECEIVER)
|
|
#if defined(CONFIG_ETRAX_RS485_DISABLE_RECEIVER)
|
|
e100_enable_rx(info);
|
|
e100_enable_rx(info);
|
|
@@ -3678,14 +3677,52 @@ rs_ioctl(struct tty_struct *tty, struct file * file,
|
|
#if defined(CONFIG_ETRAX_RS485)
|
|
#if defined(CONFIG_ETRAX_RS485)
|
|
case TIOCSERSETRS485:
|
|
case TIOCSERSETRS485:
|
|
{
|
|
{
|
|
|
|
+ /* In this ioctl we still use the old structure
|
|
|
|
+ * rs485_control for backward compatibility
|
|
|
|
+ * (if we use serial_rs485, then old user-level code
|
|
|
|
+ * wouldn't work anymore...).
|
|
|
|
+ * The use of this ioctl is deprecated: use TIOCSRS485
|
|
|
|
+ * instead.*/
|
|
struct rs485_control rs485ctrl;
|
|
struct rs485_control rs485ctrl;
|
|
|
|
+ struct serial_rs485 rs485data;
|
|
|
|
+ printk(KERN_DEBUG "The use of this ioctl is deprecated. Use TIOCSRS485 instead\n");
|
|
if (copy_from_user(&rs485ctrl, (struct rs485_control *)arg,
|
|
if (copy_from_user(&rs485ctrl, (struct rs485_control *)arg,
|
|
sizeof(rs485ctrl)))
|
|
sizeof(rs485ctrl)))
|
|
return -EFAULT;
|
|
return -EFAULT;
|
|
|
|
|
|
- return e100_enable_rs485(tty, &rs485ctrl);
|
|
|
|
|
|
+ rs485data.delay_rts_before_send = rs485ctrl.delay_rts_before_send;
|
|
|
|
+ rs485data.flags = 0;
|
|
|
|
+ if (rs485ctrl.enabled)
|
|
|
|
+ rs485data.flags |= SER_RS485_ENABLED;
|
|
|
|
+ else
|
|
|
|
+ rs485data.flags &= ~(SER_RS485_ENABLED);
|
|
|
|
+
|
|
|
|
+ if (rs485ctrl.rts_on_send)
|
|
|
|
+ rs485data.flags |= SER_RS485_RTS_ON_SEND;
|
|
|
|
+ else
|
|
|
|
+ rs485data.flags &= ~(SER_RS485_RTS_ON_SEND);
|
|
|
|
+
|
|
|
|
+ if (rs485ctrl.rts_after_sent)
|
|
|
|
+ rs485data.flags |= SER_RS485_RTS_AFTER_SEND;
|
|
|
|
+ else
|
|
|
|
+ rs485data.flags &= ~(SER_RS485_RTS_AFTER_SEND);
|
|
|
|
+
|
|
|
|
+ return e100_enable_rs485(tty, &rs485data);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+ case TIOCSRS485:
|
|
|
|
+ {
|
|
|
|
+ /* This is the new version of TIOCSRS485, with new
|
|
|
|
+ * data structure serial_rs485 */
|
|
|
|
+ struct serial_rs485 rs485data;
|
|
|
|
+ if (copy_from_user(&rs485data, (struct rs485_control *)arg,
|
|
|
|
+ sizeof(rs485data)))
|
|
|
|
+ return -EFAULT;
|
|
|
|
+
|
|
|
|
+ return e100_enable_rs485(tty, &rs485data);
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+
|
|
case TIOCSERWRRS485:
|
|
case TIOCSERWRRS485:
|
|
{
|
|
{
|
|
struct rs485_write rs485wr;
|
|
struct rs485_write rs485wr;
|
|
@@ -3827,8 +3864,8 @@ rs_close(struct tty_struct *tty, struct file * filp)
|
|
/* port closed */
|
|
/* port closed */
|
|
|
|
|
|
#if defined(CONFIG_ETRAX_RS485)
|
|
#if defined(CONFIG_ETRAX_RS485)
|
|
- if (info->rs485.enabled) {
|
|
|
|
- info->rs485.enabled = 0;
|
|
|
|
|
|
+ if (info->rs485.flags & SER_RS485_ENABLED) {
|
|
|
|
+ info->rs485.flags &= ~(SER_RS485_ENABLED);
|
|
#if defined(CONFIG_ETRAX_RS485_ON_PA)
|
|
#if defined(CONFIG_ETRAX_RS485_ON_PA)
|
|
*R_PORT_PA_DATA = port_pa_data_shadow &= ~(1 << rs485_pa_bit);
|
|
*R_PORT_PA_DATA = port_pa_data_shadow &= ~(1 << rs485_pa_bit);
|
|
#endif
|
|
#endif
|
|
@@ -4493,10 +4530,10 @@ rs_init(void)
|
|
|
|
|
|
#if defined(CONFIG_ETRAX_RS485)
|
|
#if defined(CONFIG_ETRAX_RS485)
|
|
/* Set sane defaults */
|
|
/* Set sane defaults */
|
|
- info->rs485.rts_on_send = 0;
|
|
|
|
- info->rs485.rts_after_sent = 1;
|
|
|
|
|
|
+ info->rs485.flags &= ~(SER_RS485_RTS_ON_SEND);
|
|
|
|
+ info->rs485.flags |= SER_RS485_RTS_AFTER_SEND;
|
|
info->rs485.delay_rts_before_send = 0;
|
|
info->rs485.delay_rts_before_send = 0;
|
|
- info->rs485.enabled = 0;
|
|
|
|
|
|
+ info->rs485.flags &= ~(SER_RS485_ENABLED);
|
|
#endif
|
|
#endif
|
|
INIT_WORK(&info->work, do_softint);
|
|
INIT_WORK(&info->work, do_softint);
|
|
|
|
|