forked from Minki/linux
b920de1b77
Add architecture support for the MN10300/AM33 CPUs produced by MEI to the kernel. This patch also adds board support for the ASB2303 with the ASB2308 daughter board, and the ASB2305. The only processor supported is the MN103E010, which is an AM33v2 core plus on-chip devices. [akpm@linux-foundation.org: nuke cvs control strings] Signed-off-by: Masakazu Urade <urade.masakazu@jp.panasonic.com> Signed-off-by: Koichi Yasutake <yasutake.koichi@jp.panasonic.com> Signed-off-by: David Howells <dhowells@redhat.com> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
192 lines
5.2 KiB
ArmAsm
192 lines
5.2 KiB
ArmAsm
###############################################################################
|
|
#
|
|
# Virtual DMA driver for MN10300 serial ports
|
|
#
|
|
# Copyright (C) 2007 Red Hat, Inc. All Rights Reserved.
|
|
# Written by David Howells (dhowells@redhat.com)
|
|
#
|
|
# This program is free software; you can redistribute it and/or
|
|
# modify it under the terms of the GNU General Public Licence
|
|
# as published by the Free Software Foundation; either version
|
|
# 2 of the Licence, or (at your option) any later version.
|
|
#
|
|
###############################################################################
|
|
#include <linux/sys.h>
|
|
#include <linux/linkage.h>
|
|
#include <asm/page.h>
|
|
#include <asm/smp.h>
|
|
#include <asm/cpu-regs.h>
|
|
#include <asm/frame.inc>
|
|
#include <asm/timer-regs.h>
|
|
#include <asm/proc/cache.h>
|
|
#include <asm/unit/timex.h>
|
|
#include "mn10300-serial.h"
|
|
|
|
#define SCxCTR 0x00
|
|
#define SCxICR 0x04
|
|
#define SCxTXB 0x08
|
|
#define SCxRXB 0x09
|
|
#define SCxSTR 0x0c
|
|
#define SCxTIM 0x0d
|
|
|
|
.text
|
|
|
|
###############################################################################
|
|
#
|
|
# serial port interrupt virtual DMA entry point
|
|
# - intended to run at interrupt priority 1 (not affected by local_irq_disable)
|
|
#
|
|
###############################################################################
|
|
.balign L1_CACHE_BYTES
|
|
ENTRY(mn10300_serial_vdma_interrupt)
|
|
or EPSW_IE,psw # permit overriding by
|
|
# debugging interrupts
|
|
movm [d2,d3,a2,a3,exreg0],(sp)
|
|
|
|
movhu (IAGR),a2 # see if which interrupt is
|
|
# pending
|
|
and IAGR_GN,a2
|
|
add a2,a2
|
|
add mn10300_serial_int_tbl,a2
|
|
|
|
mov (a2+),a3
|
|
mov (__iobase,a3),e2
|
|
mov (a2),a2
|
|
jmp (a2)
|
|
|
|
###############################################################################
|
|
#
|
|
# serial port receive interrupt virtual DMA entry point
|
|
# - intended to run at interrupt priority 1 (not affected by local_irq_disable)
|
|
# - stores data/status byte pairs in the ring buffer
|
|
# - induces a scheduler tick timer interrupt when done, which we then subvert
|
|
# on entry:
|
|
# A3 struct mn10300_serial_port *
|
|
# E2 I/O port base
|
|
#
|
|
###############################################################################
|
|
ENTRY(mn10300_serial_vdma_rx_handler)
|
|
mov (__rx_icr,a3),e3
|
|
mov GxICR_DETECT,d2
|
|
movbu d2,(e3) # ACK the interrupt
|
|
movhu (e3),d2 # flush
|
|
|
|
mov (__rx_inp,a3),d3
|
|
mov d3,a2
|
|
add 2,d3
|
|
and MNSC_BUFFER_SIZE-1,d3
|
|
mov (__rx_outp,a3),d2
|
|
cmp d3,d2
|
|
beq mnsc_vdma_rx_overflow
|
|
|
|
mov (__rx_buffer,a3),d2
|
|
add d2,a2
|
|
movhu (SCxSTR,e2),d2
|
|
movbu d2,(1,a2)
|
|
movbu (SCxRXB,e2),d2
|
|
movbu d2,(a2)
|
|
mov d3,(__rx_inp,a3)
|
|
bset MNSCx_RX_AVAIL,(__intr_flags,a3)
|
|
|
|
mnsc_vdma_rx_done:
|
|
mov (__tm_icr,a3),a2
|
|
mov GxICR_LEVEL_6|GxICR_ENABLE|GxICR_REQUEST|GxICR_DETECT,d2
|
|
movhu d2,(a2) # request a slow interrupt
|
|
movhu (a2),d2 # flush
|
|
|
|
movm (sp),[d2,d3,a2,a3,exreg0]
|
|
rti
|
|
|
|
mnsc_vdma_rx_overflow:
|
|
bset MNSCx_RX_OVERF,(__intr_flags,a3)
|
|
bra mnsc_vdma_rx_done
|
|
|
|
###############################################################################
|
|
#
|
|
# serial port transmit interrupt virtual DMA entry point
|
|
# - intended to run at interrupt priority 1 (not affected by local_irq_disable)
|
|
# - retrieves data bytes from the ring buffer and passes them to the serial port
|
|
# - induces a scheduler tick timer interrupt when done, which we then subvert
|
|
# A3 struct mn10300_serial_port *
|
|
# E2 I/O port base
|
|
#
|
|
###############################################################################
|
|
.balign L1_CACHE_BYTES
|
|
ENTRY(mn10300_serial_vdma_tx_handler)
|
|
mov (__tx_icr,a3),e3
|
|
mov GxICR_DETECT,d2
|
|
movbu d2,(e3) # ACK the interrupt
|
|
movhu (e3),d2 # flush
|
|
|
|
btst 0x01,(__tx_break,a3) # handle transmit break request
|
|
bne mnsc_vdma_tx_break
|
|
|
|
movbu (SCxSTR,e2),d2 # don't try and transmit a char if the
|
|
# buffer is not empty
|
|
btst SC01STR_TBF,d2 # (may have tried to jumpstart)
|
|
bne mnsc_vdma_tx_noint
|
|
|
|
movbu (__tx_xchar,a3),d2 # handle hi-pri XON/XOFF
|
|
or d2,d2
|
|
bne mnsc_vdma_tx_xchar
|
|
|
|
mov (__tx_info_buffer,a3),a2 # get the uart_info struct for Tx
|
|
mov (__xmit_tail,a2),d3
|
|
mov (__xmit_head,a2),d2
|
|
cmp d3,d2
|
|
beq mnsc_vdma_tx_empty
|
|
|
|
mov (__xmit_buffer,a2),d2 # get a char from the buffer and
|
|
# transmit it
|
|
movbu (d3,d2),d2
|
|
movbu d2,(SCxTXB,e2) # Tx
|
|
|
|
inc d3 # advance the buffer pointer
|
|
and __UART_XMIT_SIZE-1,d3
|
|
mov (__xmit_head,a2),d2
|
|
mov d3,(__xmit_tail,a2)
|
|
|
|
sub d3,d2 # see if we've written everything
|
|
beq mnsc_vdma_tx_empty
|
|
|
|
and __UART_XMIT_SIZE-1,d2 # see if we just made a hole
|
|
cmp __UART_XMIT_SIZE-2,d2
|
|
beq mnsc_vdma_tx_made_hole
|
|
|
|
mnsc_vdma_tx_done:
|
|
mov (__tm_icr,a3),a2
|
|
mov GxICR_LEVEL_6|GxICR_ENABLE|GxICR_REQUEST|GxICR_DETECT,d2
|
|
movhu d2,(a2) # request a slow interrupt
|
|
movhu (a2),d2 # flush
|
|
|
|
mnsc_vdma_tx_noint:
|
|
movm (sp),[d2,d3,a2,a3,exreg0]
|
|
rti
|
|
|
|
mnsc_vdma_tx_empty:
|
|
mov +(GxICR_LEVEL_1|GxICR_DETECT),d2
|
|
movhu d2,(e3) # disable the interrupt
|
|
movhu (e3),d2 # flush
|
|
|
|
bset MNSCx_TX_EMPTY,(__intr_flags,a3)
|
|
bra mnsc_vdma_tx_done
|
|
|
|
mnsc_vdma_tx_break:
|
|
movhu (SCxCTR,e2),d2 # turn on break mode
|
|
or SC01CTR_BKE,d2
|
|
movhu d2,(SCxCTR,e2)
|
|
mov +(GxICR_LEVEL_1|GxICR_DETECT),d2
|
|
movhu d2,(e3) # disable transmit interrupts on this
|
|
# channel
|
|
movhu (e3),d2 # flush
|
|
bra mnsc_vdma_tx_noint
|
|
|
|
mnsc_vdma_tx_xchar:
|
|
bclr 0xff,(__tx_xchar,a3)
|
|
movbu d2,(SCxTXB,e2)
|
|
bra mnsc_vdma_tx_done
|
|
|
|
mnsc_vdma_tx_made_hole:
|
|
bset MNSCx_TX_SPACE,(__intr_flags,a3)
|
|
bra mnsc_vdma_tx_done
|