Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

AMBA software DTR patch #166

Closed
wants to merge 1 commit into from
Closed
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
68 changes: 68 additions & 0 deletions drivers/tty/serial/amba-pl011.c
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,11 @@
* not have an RI input, nor do they have DTR or RTS outputs. If
* required, these have to be supplied via some other means (eg, GPIO)
* and hooked into this driver.
*
* Apparently the above paragraph was written when AMBA ports didn't
* have DTR. The code below shows that some AMBA ports DO have a DTR,
* but others do not. The DTR through GPIO implementation is now
* present.
*/

#if defined(CONFIG_SERIAL_AMBA_PL011_CONSOLE) && defined(CONFIG_MAGIC_SYSRQ)
Expand All @@ -51,10 +56,15 @@
#include <linux/dma-mapping.h>
#include <linux/scatterlist.h>
#include <linux/delay.h>
#include <linux/gpio.h>

#include <asm/io.h>
#include <asm/sizes.h>



// The maximum number of amba ports supported. Increase this if you need
// more (unlikely). -- REW
#define UART_NR 14

#define SERIAL_AMBA_MAJOR 204
Expand Down Expand Up @@ -162,6 +172,7 @@ struct uart_amba_port {
bool autorts;
char type[12];
bool interrupt_may_hang; /* vendor-specific */
int dtrpin;
#ifdef CONFIG_DMA_ENGINE
/* DMA stuff */
bool using_tx_dma;
Expand All @@ -171,6 +182,18 @@ struct uart_amba_port {
#endif
};


static int dtr_pin[UART_NR];
static int gpio_inited;

// We need to know the number of passed software-dtr-gpio-numbers.
// Otherwise it would be impossible to distiguish between "not specified"
// and specified as GPIO-0.
static int num_dtrs;

module_param_array (dtr_pin, int, &num_dtrs, 0);


/*
* Reads up to 256 characters from the FIFO or until it's empty and
* inserts them into the TTY layer. Returns the number of characters
Expand Down Expand Up @@ -1305,6 +1328,8 @@ static void pl011_set_mctrl(struct uart_port *port, unsigned int mctrl)
cr &= ~uartbit

TIOCMBIT(TIOCM_RTS, UART011_CR_RTS);
// On the raspberry pi this writes the bit documented as:
// "DTR: unsupported, write zero". Oh well.. -- REW
TIOCMBIT(TIOCM_DTR, UART011_CR_DTR);
TIOCMBIT(TIOCM_OUT1, UART011_CR_OUT1);
TIOCMBIT(TIOCM_OUT2, UART011_CR_OUT2);
Expand All @@ -1316,6 +1341,17 @@ static void pl011_set_mctrl(struct uart_port *port, unsigned int mctrl)
}
#undef TIOCMBIT

// Software (GPIO) DTR support.
if (uap->dtrpin >= 0) {
// Apparently, this is called without us having been able
// to grab the gpio pin. So we'll have to check the flag.
// (I'm assuming this is called early during boot, as I don't
// see this happening when the system is up. That's why a
// global static flag works).
if (gpio_inited)
gpio_set_value (uap->dtrpin, (mctrl & TIOCM_DTR )?0:1);
}

writew(cr, uap->port.membase + UART011_CR);
}

Expand Down Expand Up @@ -1449,6 +1485,21 @@ static int pl011_startup(struct uart_port *port)
plat->init();
}

// Software (GPIO) DTR support.
if (uap->dtrpin >= 0) {
retval = gpio_request (uap->dtrpin, "pl011.dtr");
if (retval < 0) {
printk ("DTR request GPIO failed. \n");
goto clk_dis;
}
retval = gpio_direction_output(uap->dtrpin, 0);
if (retval < 0) {
printk ("DTR GPIO output failed. \n");
goto clk_dis;
}
gpio_inited = 1;
}

return 0;

clk_dis:
Expand Down Expand Up @@ -1488,6 +1539,15 @@ static void pl011_shutdown(struct uart_port *port)
* Free the interrupt
*/
free_irq(uap->port.irq, uap);

if (uap->dtrpin >= 0) {
// I considered making the pin input again. However, when
// software DTR is configured, there is most likely hardware
// attached, so we better keep driving the pin, probably in
// the "not-active" state.
gpio_free (uap->dtrpin);
gpio_inited = 0;
}

/*
* disable the port
Expand Down Expand Up @@ -1919,6 +1979,14 @@ static int pl011_probe(struct amba_device *dev, const struct amba_id *id)
goto unmap;
}


// Specify -1 for no software DTR (to skip over ports).
// Specify anything else to use that GPIO.
if (i < num_dtrs)
uap->dtrpin = dtr_pin[i];
else
uap->dtrpin = -1;

uap->vendor = vendor;
uap->lcrh_rx = vendor->lcrh_rx;
uap->lcrh_tx = vendor->lcrh_tx;
Expand Down