/*
 * Copyright (c) 2010, Mariano Alvira <mar@devl.org> and other contributors
 * to the MC1322x project (http://mc1322x.devl.org)
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 * 1. Redistributions of source code must retain the above copyright
 *    notice, this list of conditions and the following disclaimer.
 * 2. Redistributions in binary form must reproduce the above copyright
 *    notice, this list of conditions and the following disclaimer in the
 *    documentation and/or other materials provided with the distribution.
 * 3. Neither the name of the Institute nor the names of its contributors
 *    may be used to endorse or promote products derived from this software
 *    without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE INSTITUTE AND CONTRIBUTORS ``AS IS'' AND
 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 * ARE DISCLAIMED.  IN NO EVENT SHALL THE INSTITUTE OR CONTRIBUTORS BE LIABLE
 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
 * SUCH DAMAGE.
 *
 * This file is part of libmc1322x: see http://mc1322x.devl.org
 * for details. 
 *
 *
 */

#include <mc1322x.h>
#include <stdint.h>

volatile char u2_tx_buf[UART2_TX_BUFFERSIZE];
volatile uint32_t u2_tx_head, u2_tx_tail;

#if UART2_RX_BUFFERSIZE > 32
volatile char u2_rx_buf[UART2_RX_BUFFERSIZE-32];
volatile uint32_t u2_rx_head, u2_rx_tail;
#endif

void uart2_isr(void) {

#if UART2_RX_BUFFERSIZE > 32
	if (*UART2_USTAT & ( 1 << 6)) {   //receive interrupt
		while( *UART2_URXCON != 0 ) {   //flush the hardware fifo into the software buffer
			uint32_t u2_rx_tail_next;
			u2_rx_tail_next = u2_rx_tail+1;
			if (u2_rx_tail_next >= sizeof(u2_rx_buf))
				u2_rx_tail_next = 0;
			if (u2_rx_head != u2_rx_tail_next) {
				u2_rx_buf[u2_rx_tail]= *UART2_UDATA;
				u2_rx_tail =  u2_rx_tail_next;
			} else {  //buffer is full, flush the fifo
				while (*UART2_URXCON !=0) { if (*UART2_UDATA) { } }
			}
		}
		return;
	}
#endif

	while( *UART2_UTXCON != 0 ) {
		if (u2_tx_head == u2_tx_tail) {
#if UART2_RX_BUFFERSIZE > 32
            *UART2_UCON |= (1 << 13); /*disable tx interrupt */
#else
			disable_irq(UART2);
#endif
			return;
		}
		*UART2_UDATA = u2_tx_buf[u2_tx_tail];
		u2_tx_tail++;
		if (u2_tx_tail >= sizeof(u2_tx_buf))
			u2_tx_tail = 0;
	}

}

void uart2_putc(char c) {
	/* disable UART2 since */
	/* UART2 isr modifies u2_tx_head and u2_tx_tail */
#if UART2_RX_BUFFERSIZE > 32
            *UART2_UCON |= (1 << 13); /*disable tx interrupt */
#else
			disable_irq(UART2);
#endif

	if( (u2_tx_head == u2_tx_tail) &&
	    (*UART2_UTXCON != 0)) {
		*UART2_UDATA = c;
	} else {
		u2_tx_buf[u2_tx_head] = c;
		u2_tx_head += 1;
		if (u2_tx_head >= sizeof(u2_tx_buf))
			u2_tx_head = 0;
		if (u2_tx_head == u2_tx_tail) { /* drop chars when no room */
#if UART2_DROP_CHARS
			if (u2_tx_head) { u2_tx_head -=1; } else { u2_tx_head = sizeof(u2_tx_buf); }
#else
			{
				uint32_t  u2_tx_tail_save=u2_tx_tail;
                                /* Back up head to show buffer not empty, and enable tx interrupt */
				u2_tx_head--;
#if UART2_RX_BUFFERSIZE > 32
				*UART2_UCON &= ~(1 << 13); /*enable tx interrupt */
#else
				enable_irq(UART2);
#endif
                                /* Tail will change after one character goes out */
				while (u2_tx_tail_save == u2_tx_tail) ;
                                /* Restore head to character we just stuffed */
				u2_tx_head++;
				return;
			}
#endif /* UART2_DROP_CHARS */
		}

#if UART2_RX_BUFFERSIZE > 32
		*UART2_UCON &= ~(1 << 13); /*enable tx interrupt */
#else
		enable_irq(UART2);
#endif

	}
}

uint8_t uart2_getc(void) {
#if UART2_RX_BUFFERSIZE > 32
/* First pull from the ram buffer */
uint8_t c=0;
  if (u2_rx_head != u2_rx_tail) {
    c = u2_rx_buf[u2_rx_head++];
    if (u2_rx_head >= sizeof(u2_rx_buf))
        u2_rx_head=0;
    return c;
  }
#endif
/* Then pull from the hardware fifo */
	while(uart2_can_get() == 0) { continue; }
	return *UART2_UDATA;
}