Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in
Toggle navigation
L
linux-davinci
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Redmine
Redmine
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Operations
Operations
Metrics
Environments
Analytics
Analytics
CI / CD
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
linux
linux-davinci
Commits
70ca65ca
Commit
70ca65ca
authored
May 09, 2005
by
Tony Lindgren
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Add OMAP IrDA driver
Adds OMAP IrDA driver Signed-off-by:
Tony Lindgren
<
tony@atomide.com
>
parent
5e97cfa1
Changes
3
Hide whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
1018 additions
and
0 deletions
+1018
-0
drivers/net/irda/Kconfig
drivers/net/irda/Kconfig
+6
-0
drivers/net/irda/Makefile
drivers/net/irda/Makefile
+2
-0
drivers/net/irda/omap1610-ir.c
drivers/net/irda/omap1610-ir.c
+1010
-0
No files found.
drivers/net/irda/Kconfig
View file @
70ca65ca
...
...
@@ -341,6 +341,12 @@ config TOSHIBA_FIR
To compile it as a module, choose M here: the module will be called
donauboe.
config OMAP1610_IR
tristate "OMAP1610 IrDA(SIR/MIR/FIR)"
depends on IRDA
help
Say Y here if you want to build support for the Omap1610 IR.
config AU1000_FIR
tristate "Alchemy Au1000 SIR/FIR"
depends on MIPS_AU1000 && IRDA
...
...
drivers/net/irda/Makefile
View file @
70ca65ca
...
...
@@ -42,6 +42,8 @@ obj-$(CONFIG_OLD_BELKIN_DONGLE) += old_belkin-sir.o
obj-$(CONFIG_MCP2120_DONGLE)
+=
mcp2120-sir.o
obj-$(CONFIG_ACT200L_DONGLE)
+=
act200l-sir.o
obj-$(CONFIG_MA600_DONGLE)
+=
ma600-sir.o
obj-$(CONFIG_OMAP1610_IR)
+=
omap1610-ir.o
# The SIR helper module
sir-dev-objs
:=
sir_core.o sir_dev.o sir_dongle.o sir_kthread.o
drivers/net/irda/omap1610-ir.c
0 → 100644
View file @
70ca65ca
/*
* BRIEF MODULE DESCRIPTION
*
* Infra-red driver for the OMAP1610-H2 and OMAP1710-H3 Platforms
* (SIR/MIR/FIR modes)
* (based on omap-sir.c)
*
* Copyright 2003 MontaVista Software Inc.
* Author: MontaVista Software, Inc.
* source@mvista.com
*
* Copyright 2004 Texas Instruments.
*
* This program is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License as published by the
* Free Software Foundation; either version 2 of the License, or (at your
* option) any later version.
*
* THIS SOFTWARE IS PROVIDED ``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 AUTHOR 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.
*
* You should have received a copy of the GNU General Public License along
* with this program; if not, write to the Free Software Foundation, Inc.,
* 675 Mass Ave, Cambridge, MA 02139, USA.
*
Modifications:
Feb 2004, Texas Instruments
- Ported to 2.6 kernel (Feb 2004).
*
Apr 2004, Texas Instruments
- Added support for H3 (Apr 2004).
Nov 2004, Texas Instruments
- Added support for Power Management.
*/
#include <linux/config.h>
#include <linux/module.h>
#include <linux/types.h>
#include <linux/init.h>
#include <linux/errno.h>
#include <linux/netdevice.h>
#include <linux/slab.h>
#include <linux/rtnetlink.h>
#include <linux/interrupt.h>
#include <linux/delay.h>
#include <linux/ioport.h>
#include <linux/dma-mapping.h>
#include <net/irda/irda.h>
#include <net/irda/irmod.h>
#include <net/irda/wrapper.h>
#include <net/irda/irda_device.h>
#include <asm/irq.h>
#include <asm/io.h>
#include <asm/hardware.h>
#include <asm/serial.h>
#include <asm/mach-types.h>
#include <asm/dma.h>
#include <asm/arch/mux.h>
#include <asm/arch/gpio.h>
#include <linux/i2c.h>
#ifdef CONFIG_MACH_OMAP_H3
#include <asm/arch/gpioexpander.h>
#endif
#define SIR_MODE 0
#define MIR_MODE 1
#define FIR_MODE 2
#define OMAP1610_H2_FIRSEL_GPIO 17
static
int
rx_state
=
0
;
/* RX state for IOCTL */
struct
omap1610_irda
{
unsigned
char
open
;
int
speed
;
/* Current IrDA speed */
int
newspeed
;
struct
net_device_stats
stats
;
struct
irlap_cb
*
irlap
;
struct
qos_info
qos
;
int
rx_dma_channel
;
int
tx_dma_channel
;
dma_addr_t
rx_buf_dma_phys
;
/* Physical adress of RX DMA buffer */
dma_addr_t
tx_buf_dma_phys
;
/* Physical adress of TX DMA buffer */
void
*
rx_buf_dma_virt
;
/* Virtual adress of RX DMA buffer */
void
*
tx_buf_dma_virt
;
/* Virtual adress of TX DMA buffer */
struct
device
*
dev
;
};
#define OMAP_IRDA_DEBUG 0
#if (OMAP_IRDA_DEBUG > 0)
#define DBG(format, args...) printk(KERN_ERR "%s(): " format, __FUNCTION__, ## args);
#define DBG_IRQ(format, args...) printk(KERN_ERR "%s(): " format, __FUNCTION__, ## args);
#else
#define DBG(format, args...)
#define DBG_IRQ(format, args...)
#endif
#if (OMAP_IRDA_DEBUG > 1)
#define __ECHO_IN printk(KERN_ERR "%s: enter\n",__FUNCTION__);
#define __ECHO_OUT printk(KERN_ERR "%s: exit\n",__FUNCTION__);
#else
#define __ECHO_IN
#define __ECHO_OUT
#endif
#ifdef OMAP1610_IR_HARDWARE_DEBUG_ENABLE
#define HDBG_DELAY 200
void
hard_debug1
(
u16
i
)
{
for
(;
i
;
i
--
)
{
omap_writew
(
0x2000
,
OMAP1610_GPIO1_BASE
+
OMAP1610_GPIO_CLEAR_DATAOUT
);
udelay
(
HDBG_DELAY
);
omap_writew
(
0x2000
,
OMAP1610_GPIO1_BASE
+
OMAP1610_GPIO_SET_DATAOUT
);
udelay
(
HDBG_DELAY
);
}
}
void
hard_debug2
(
u16
i
)
{
for
(;
i
;
i
--
)
{
omap_writew
(
0x8000
,
OMAP1610_GPIO1_BASE
+
OMAP1610_GPIO_CLEAR_DATAOUT
);
udelay
(
HDBG_DELAY
);
omap_writew
(
0x8000
,
OMAP1610_GPIO1_BASE
+
OMAP1610_GPIO_SET_DATAOUT
);
udelay
(
HDBG_DELAY
);
}
}
#define HDBG1(i) hard_debug1(i)
#define HDBG2(i) hard_debug2(i)
#else
#define HDBG1(i)
#define HDBG2(i)
#endif
/* forward declarations */
extern
void
irda_device_setup
(
struct
net_device
*
dev
);
extern
void
omap_stop_dma
(
int
lch
);
static
int
omap1610_irda_set_speed
(
struct
net_device
*
dev
,
int
speed
);
static
void
omap1610_irda_start_rx_dma
(
struct
omap1610_irda
*
si
)
{
/* Configure DMA */
omap_set_dma_src_params
(
si
->
rx_dma_channel
,
0x3
,
0x0
,
(
unsigned
long
)
UART3_RHR
);
omap_enable_dma_irq
(
si
->
rx_dma_channel
,
0x01
);
omap_set_dma_dest_params
(
si
->
rx_dma_channel
,
0x0
,
0x1
,
si
->
rx_buf_dma_phys
);
omap_set_dma_transfer_params
(
si
->
rx_dma_channel
,
0x0
,
4096
,
0x1
,
0x0
);
omap_start_dma
(
si
->
rx_dma_channel
);
}
static
void
omap1610_start_tx_dma
(
struct
omap1610_irda
*
si
,
int
size
)
{
__ECHO_IN
;
/* Configure DMA */
omap_set_dma_dest_params
(
si
->
tx_dma_channel
,
0x03
,
0x0
,
(
unsigned
long
)
UART3_THR
);
omap_enable_dma_irq
(
si
->
tx_dma_channel
,
0x01
);
omap_set_dma_src_params
(
si
->
tx_dma_channel
,
0x0
,
0x1
,
si
->
tx_buf_dma_phys
);
omap_set_dma_transfer_params
(
si
->
tx_dma_channel
,
0x0
,
size
,
0x1
,
0x0
);
HDBG1
(
1
);
/* Start DMA */
omap_start_dma
(
si
->
tx_dma_channel
);
HDBG1
(
1
);
__ECHO_OUT
;
}
/* DMA RX callback - normally, we should not go here,
it calls only if something is going wrong
*/
static
void
omap1610_irda_rx_dma_callback
(
int
lch
,
u16
ch_status
,
void
*
data
)
{
struct
net_device
*
dev
=
data
;
struct
omap1610_irda
*
si
=
dev
->
priv
;
printk
(
KERN_ERR
"RX Transfer error or very big frame
\n
"
);
/* Clear interrupts */
omap_readb
(
UART3_IIR
);
si
->
stats
.
rx_frame_errors
++
;
omap_readb
(
UART3_RESUME
);
/* Re-init RX DMA */
omap1610_irda_start_rx_dma
(
si
);
}
/* DMA TX callback - calling when frame transfer has been finished */
static
void
omap1610_irda_tx_dma_callback
(
int
lch
,
u16
ch_status
,
void
*
data
)
{
struct
net_device
*
dev
=
data
;
struct
omap1610_irda
*
si
=
dev
->
priv
;
__ECHO_IN
;
/*Stop DMA controller */
omap_stop_dma
(
si
->
tx_dma_channel
);
__ECHO_OUT
;
}
/*
* Set the IrDA communications speed.
* Interrupt have to be disabled here.
*/
static
int
omap1610_irda_startup
(
struct
net_device
*
dev
)
{
__ECHO_IN
;
/* Enable UART3 clock and set UART3 to IrDA mode */
omap_writel
(
omap_readl
(
MOD_CONF_CTRL_0
)
|
(
1
<<
31
)
|
(
1
<<
15
),
MOD_CONF_CTRL_0
);
if
(
machine_is_omap_h2
())
{
// omap_cfg_reg(Y15_1610_GPIO17);
omap_writel
(
omap_readl
(
FUNC_MUX_CTRL_A
)
|
7
,
FUNC_MUX_CTRL_A
);
omap_set_gpio_direction
(
OMAP1610_H2_FIRSEL_GPIO
,
0
);
omap_set_gpio_dataout
(
OMAP1610_H2_FIRSEL_GPIO
,
0
);
}
omap_writeb
(
0x07
,
UART3_MDR1
);
/* Put UART3 in reset mode */
/* Clear DLH and DLL */
omap_writeb
(
1
<<
7
,
UART3_LCR
);
omap_writeb
(
0
,
UART3_DLL
);
omap_writeb
(
0
,
UART3_DLH
);
omap_writeb
(
0xbf
,
UART3_LCR
);
omap_writeb
(
1
<<
4
,
UART3_EFR
);
omap_writeb
(
1
<<
7
,
UART3_LCR
);
/* Enable access to UART3_TLR and UART3_TCR registers */
omap_writeb
(
1
<<
6
,
UART3_MCR
);
omap_writeb
(
0
,
UART3_SCR
);
/* Set Rx trigger to 1 and Tx trigger to 1 */
omap_writeb
(
0
,
UART3_TLR
);
/* Set LCR to 8 bits and 1 stop bit */
omap_writeb
(
0x03
,
UART3_LCR
);
/* Clear RX and TX FIFO and enable FIFO */
/* Use DMA Req for transfers */
omap_writeb
((
1
<<
2
)
|
(
1
<<
1
)
|
(
1
<<
3
)
|
(
1
<<
4
)
|
(
1
<<
6
)
|
1
,
UART3_FCR
);
omap_writeb
(
0
,
UART3_MCR
);
omap_writeb
((
1
<<
7
)
|
(
1
<<
6
),
UART3_SCR
);
/* Enable UART3 SIR Mode,(Frame-length method to end frames) */
omap_writeb
(
1
,
UART3_MDR1
);
/* Set Status FIFO trig to 1 */
omap_writeb
(
0
,
UART3_MDR2
);
/* Enables RXIR input */
/* and disable TX underrun */
/* SEND_SIP pulse */
// omap_writeb((1 << 7) | (1 << 6) | (1 << 4), UART3_ACREG);
omap_writeb
((
1
<<
6
)
|
(
1
<<
4
),
UART3_ACREG
);
/* Enable EOF Interrupt only */
omap_writeb
((
1
<<
7
)
|
(
1
<<
5
),
UART3_IER
);
/* Set Maximum Received Frame size to 2048 bytes */
omap_writeb
(
0x00
,
UART3_RXFLL
);
omap_writeb
(
0x08
,
UART3_RXFLH
);
omap_readb
(
UART3_RESUME
);
__ECHO_OUT
;
return
0
;
}
static
int
omap1610_irda_shutdown
(
struct
omap1610_irda
*
si
)
{
/* Disable all UART3 Interrupts */
omap_writeb
(
0
,
UART3_IER
);
/* Disable UART3 and disable baud rate generator */
omap_writeb
(
0x07
,
UART3_MDR1
);
/* Put UART3 in reset mode */
omap_writeb
((
1
<<
5
),
UART3_ACREG
);
/* set SD_MODE pin to high and Disable RX IR */
/* Clear DLH and DLL */
omap_writeb
(
1
<<
7
,
UART3_LCR
);
omap_writeb
(
0
,
UART3_DLL
);
omap_writeb
(
0
,
UART3_DLH
);
return
0
;
}
static
irqreturn_t
omap1610_irda_irq
(
int
irq
,
void
*
dev_id
,
struct
pt_regs
*
hw_regs
)
{
struct
net_device
*
dev
=
dev_id
;
struct
omap1610_irda
*
si
=
dev
->
priv
;
struct
sk_buff
*
skb
;
u8
status
;
int
w
=
0
;
__ECHO_IN
;
/* Clear EOF interrupt */
status
=
omap_readb
(
UART3_IIR
);
if
(
status
&
(
1
<<
5
))
{
u8
mdr2
=
omap_readb
(
UART3_MDR2
);
HDBG1
(
2
);
if
(
mdr2
&
1
)
printk
(
KERN_ERR
"IRDA Buffer underrun error"
);
si
->
stats
.
tx_packets
++
;
if
(
si
->
newspeed
)
{
omap1610_irda_set_speed
(
dev
,
si
->
newspeed
);
si
->
newspeed
=
0
;
}
netif_wake_queue
(
dev
);
if
(
!
(
status
&
0x80
))
return
IRQ_HANDLED
;
}
/* Stop DMA and if there are no errors, send frame to upper layer */
omap_stop_dma
(
si
->
rx_dma_channel
);
status
=
omap_readb
(
UART3_SFLSR
);
/* Take a frame status */
if
(
status
!=
0
)
{
/* Bad frame? */
si
->
stats
.
rx_frame_errors
++
;
omap_readb
(
UART3_RESUME
);
}
else
{
/* We got a frame! */
skb
=
alloc_skb
(
4096
,
GFP_ATOMIC
);
if
(
!
skb
)
{
printk
(
KERN_ERR
"omap_sir: out of memory for RX SKB
\n
"
);
return
IRQ_HANDLED
;
}
/*
* Align any IP headers that may be contained
* within the frame.
*/
skb_reserve
(
skb
,
1
);
w
=
omap_readw
(
OMAP_DMA_CDAC
(
si
->
rx_dma_channel
));
w
-=
omap_readw
(
OMAP_DMA_CDSA_L
(
si
->
rx_dma_channel
));
if
(
si
->
speed
!=
4000000
)
{
memcpy
(
skb_put
(
skb
,
w
-
2
),
si
->
rx_buf_dma_virt
,
w
-
2
);
/* Copy DMA buffer to skb */
}
else
{
memcpy
(
skb_put
(
skb
,
w
-
4
),
si
->
rx_buf_dma_virt
,
w
-
4
);
/* Copy DMA buffer to skb */
}
skb
->
dev
=
dev
;
skb
->
mac
.
raw
=
skb
->
data
;
skb
->
protocol
=
htons
(
ETH_P_IRDA
);
si
->
stats
.
rx_packets
++
;
si
->
stats
.
rx_bytes
+=
skb
->
len
;
netif_receive_skb
(
skb
);
/* Send data to upper level */
}
/* Re-init RX DMA */
omap1610_irda_start_rx_dma
(
si
);
dev
->
last_rx
=
jiffies
;
__ECHO_OUT
;
return
IRQ_HANDLED
;
}
static
int
omap1610_irda_hard_xmit
(
struct
sk_buff
*
skb
,
struct
net_device
*
dev
)
{
struct
omap1610_irda
*
si
=
dev
->
priv
;
int
speed
=
irda_get_next_speed
(
skb
);
int
mtt
=
irda_get_mtt
(
skb
);
int
xbofs
=
irda_get_next_xbofs
(
skb
);
__ECHO_IN
;
/*
* Does this packet contain a request to change the interface
* speed? If so, remember it until we complete the transmission
* of this frame.
*/
if
(
speed
!=
si
->
speed
&&
speed
!=
-
1
)
si
->
newspeed
=
speed
;
if
(
xbofs
)
{
/* Set number of addtional BOFS */
omap_writeb
(
xbofs
+
1
,
UART3_EBLR
);
}
/*
* If this is an empty frame, we can bypass a lot.
*/
if
(
skb
->
len
==
0
)
{
if
(
si
->
newspeed
)
{
si
->
newspeed
=
0
;
omap1610_irda_set_speed
(
dev
,
speed
);
}
dev_kfree_skb
(
skb
);
return
0
;
}
netif_stop_queue
(
dev
);
/* Copy skb data to DMA buffer */
memcpy
(
si
->
tx_buf_dma_virt
,
skb
->
data
,
skb
->
len
);
si
->
stats
.
tx_bytes
+=
skb
->
len
;
/* Set frame length */
omap_writeb
((
skb
->
len
&
0xff
),
UART3_TXFLL
);
omap_writeb
((
skb
->
len
>>
8
),
UART3_TXFLH
);
if
(
mtt
>
1000
)
mdelay
(
mtt
/
1000
);
else
udelay
(
mtt
);
/* Start TX DMA transfer */
omap1610_start_tx_dma
(
si
,
skb
->
len
);
/* We can free skb now because it's already in DMA buffer */
dev_kfree_skb
(
skb
);
dev
->
trans_start
=
jiffies
;
__ECHO_OUT
;
return
0
;
}
static
int
omap1610_irda_ioctl
(
struct
net_device
*
dev
,
struct
ifreq
*
ifreq
,
int
cmd
)
{
struct
if_irda_req
*
rq
=
(
struct
if_irda_req
*
)
ifreq
;
struct
omap1610_irda
*
si
=
dev
->
priv
;
int
ret
=
-
EOPNOTSUPP
;
__ECHO_IN
;
switch
(
cmd
)
{
case
SIOCSBANDWIDTH
:
if
(
capable
(
CAP_NET_ADMIN
))
{
/*
* We are unable to set the speed if the
* device is not running.
*/
if
(
si
->
open
)
{
ret
=
omap1610_irda_set_speed
(
dev
,
rq
->
ifr_baudrate
);
}
else
{
printk
(
KERN_ERR
"omap_irda_ioctl: SIOCSBANDWIDTH: !netif_running
\n
"
);
ret
=
0
;
}
}
break
;
case
SIOCSMEDIABUSY
:
ret
=
-
EPERM
;
if
(
capable
(
CAP_NET_ADMIN
))
{
irda_device_set_media_busy
(
dev
,
TRUE
);
ret
=
0
;
}
break
;
case
SIOCGRECEIVING
:
rq
->
ifr_receiving
=
rx_state
;
break
;
default:
break
;
}
__ECHO_OUT
;
return
ret
;
}
static
struct
net_device_stats
*
omap1610_irda_stats
(
struct
net_device
*
dev
)
{
struct
omap1610_irda
*
si
=
dev
->
priv
;
return
&
si
->
stats
;
}
static
int
omap1610_irda_start
(
struct
net_device
*
dev
)
{
struct
omap1610_irda
*
si
=
dev
->
priv
;
int
err
;
unsigned
long
flags
=
0
;
#ifdef CONFIG_MACH_OMAP_H3
u8
ioExpanderVal
=
0
;
#endif
__ECHO_IN
;
si
->
speed
=
9600
;
err
=
request_irq
(
dev
->
irq
,
omap1610_irda_irq
,
0
,
dev
->
name
,
dev
);
if
(
err
)
goto
err_irq
;
/*
* The interrupt must remain disabled for now.
*/
disable_irq
(
dev
->
irq
);
/* Request DMA channels for IrDA hardware */
if
(
omap_request_dma
(
OMAP_DMA_UART3_RX
,
"IrDA Rx DMA"
,
(
void
*
)
omap1610_irda_rx_dma_callback
,
dev
,
&
(
si
->
rx_dma_channel
)))
{
printk
(
KERN_ERR
"Failed to request IrDA Rx DMA
\n
"
);
goto
err_irq
;
}
if
(
omap_request_dma
(
OMAP_DMA_UART3_TX
,
"IrDA Tx DMA"
,
(
void
*
)
omap1610_irda_tx_dma_callback
,
dev
,
&
(
si
->
tx_dma_channel
)))
{
printk
(
KERN_ERR
"Failed to request IrDA Tx DMA
\n
"
);
goto
err_irq
;
}
/* Allocate TX and RX buffers for DMA channels */
si
->
rx_buf_dma_virt
=
dma_alloc_coherent
(
NULL
,
4096
,
&
(
si
->
rx_buf_dma_phys
),
flags
);
si
->
tx_buf_dma_virt
=
dma_alloc_coherent
(
NULL
,
4096
,
&
(
si
->
tx_buf_dma_phys
),
flags
);
/*
* Setup the serial port for the specified config.
*/
#if CONFIG_MACH_OMAP_H3
if
((
err
=
read_gpio_expa
(
&
ioExpanderVal
,
0x26
)))
{
printk
(
KERN_ERR
"Error reading from I/O EXPANDER
\n
"
);
return
err
;
}
ioExpanderVal
|=
0x40
;
/* 'P6' Enable IRDA_TX and IRDA_RX */
if
((
err
=
write_gpio_expa
(
ioExpanderVal
,
0x26
)))
{
printk
(
KERN_ERR
"Error writing to I/O EXPANDER
\n
"
);
return
err
;
}
#endif
err
=
omap1610_irda_startup
(
dev
);
if
(
err
)
goto
err_startup
;
omap1610_irda_set_speed
(
dev
,
si
->
speed
=
9600
);
/*
* Open a new IrLAP layer instance.
*/
si
->
irlap
=
irlap_open
(
dev
,
&
si
->
qos
,
"omap_sir"
);
err
=
-
ENOMEM
;
if
(
!
si
->
irlap
)
goto
err_irlap
;
/* Now enable the interrupt and start the queue */
si
->
open
=
1
;
/* Start RX DMA */
omap1610_irda_start_rx_dma
(
si
);
enable_irq
(
dev
->
irq
);
netif_start_queue
(
dev
);
__ECHO_OUT
;
return
0
;
err_irlap:
si
->
open
=
0
;
omap1610_irda_shutdown
(
si
);
err_startup:
err_irq:
free_irq
(
dev
->
irq
,
dev
);
return
err
;
}
static
int
omap1610_irda_stop
(
struct
net_device
*
dev
)
{
struct
omap1610_irda
*
si
=
dev
->
priv
;
__ECHO_IN
;
disable_irq
(
dev
->
irq
);
netif_stop_queue
(
dev
);
omap_free_dma
(
si
->
rx_dma_channel
);
omap_free_dma
(
si
->
tx_dma_channel
);
dma_free_coherent
(
NULL
,
4096
,
si
->
rx_buf_dma_virt
,
si
->
rx_buf_dma_phys
);
dma_free_coherent
(
NULL
,
4096
,
si
->
tx_buf_dma_virt
,
si
->
tx_buf_dma_phys
);
omap1610_irda_shutdown
(
si
);
/* Stop IrLAP */
if
(
si
->
irlap
)
{
irlap_close
(
si
->
irlap
);
si
->
irlap
=
NULL
;
}
si
->
open
=
0
;
/*
* Free resources
*/
free_irq
(
dev
->
irq
,
dev
);
__ECHO_OUT
;
return
0
;
}
#ifdef CONFIG_MACH_OMAP_H3
static
void
set_h3_gpio_expa
(
u8
FIR_SEL
,
u8
IrDA_INVSEL
)
{
u8
ioExpanderVal
=
0
;
if
(
read_gpio_expa
(
&
ioExpanderVal
,
0x27
)
!=
0
)
{
printk
(
KERN_ERR
"Error reading from I/O EXPANDER
\n
"
);
return
;
}
ioExpanderVal
&=
~
0x03
;
ioExpanderVal
|=
FIR_SEL
<<
1
;
ioExpanderVal
|=
IrDA_INVSEL
<<
0
;
if
(
write_gpio_expa
(
ioExpanderVal
,
0x27
)
!=
0
)
{
printk
(
KERN_ERR
"Error writing to I/O EXPANDER
\n
"
);
return
;
}
if
(
read_gpio_expa
(
&
ioExpanderVal
,
0x27
)
!=
0
)
{
printk
(
KERN_ERR
"Error reading from I/O EXPANDER
\n
"
);
return
;
}
}
int
which_speed
;
static
void
set_h3_gpio_expa_handler
(
void
*
data
)
{
int
*
mode
=
data
;
if
(
*
mode
==
SIR_MODE
)
set_h3_gpio_expa
(
0
,
1
);
else
if
(
*
mode
==
MIR_MODE
)
set_h3_gpio_expa
(
1
,
1
);
else
if
(
*
mode
==
FIR_MODE
)
set_h3_gpio_expa
(
1
,
1
);
}
DECLARE_WORK
(
set_h3_gpio_expa_work
,
&
set_h3_gpio_expa_handler
,
&
which_speed
);
static
inline
void
set_h3_irda_mode
(
int
mode
)
{
cancel_delayed_work
(
&
set_h3_gpio_expa_work
);
which_speed
=
mode
;
schedule_work
(
&
set_h3_gpio_expa_work
);
}
#else
#define set_h3_irda_mode(x)
#endif
static
int
omap1610_irda_set_speed
(
struct
net_device
*
dev
,
int
speed
)
{
struct
omap1610_irda
*
si
=
dev
->
priv
;
int
divisor
;
__ECHO_IN
;
/* Set IrDA speed */
if
(
speed
<=
115200
)
{
/* SIR mode */
if
(
machine_is_omap_h2
())
{
omap_set_gpio_dataout
(
OMAP1610_H2_FIRSEL_GPIO
,
0
);
}
if
(
machine_is_omap_h3
())
set_h3_irda_mode
(
SIR_MODE
);
printk
(
"Set SIR Mode! Speed: %d
\n
"
,
speed
);
omap_writeb
(
1
,
UART3_MDR1
);
/* Set SIR mode */
omap_writeb
(
1
,
UART3_EBLR
);
divisor
=
48000000
/
(
16
*
speed
);
/* Base clock 48 MHz */
HDBG2
(
1
);
omap_writeb
(
1
<<
7
,
UART3_LCR
);
omap_writeb
((
divisor
&
0xFF
),
UART3_DLL
);
omap_writeb
((
divisor
>>
8
),
UART3_DLH
);
omap_writeb
(
0x03
,
UART3_LCR
);
omap_writeb
(
0
,
UART3_MCR
);
HDBG2
(
1
);
}
else
if
(
speed
<=
1152000
)
{
/* MIR mode */
printk
(
"Set MIR Mode! Speed: %d
\n
"
,
speed
);
omap_writeb
((
1
<<
2
)
|
(
1
<<
6
),
UART3_MDR1
);
/* Set MIR mode with
SIP after each frame */
omap_writeb
(
2
,
UART3_EBLR
);
divisor
=
48000000
/
(
41
*
speed
);
/* Base clock 48 MHz */
omap_writeb
(
1
<<
7
,
UART3_LCR
);
omap_writeb
((
divisor
&
0xFF
),
UART3_DLL
);
omap_writeb
((
divisor
>>
8
),
UART3_DLH
);
omap_writeb
(
0x03
,
UART3_LCR
);
if
(
machine_is_omap_h2
())
omap_set_gpio_dataout
(
OMAP1610_H2_FIRSEL_GPIO
,
1
);
if
(
machine_is_omap_h3
())
set_h3_irda_mode
(
MIR_MODE
);
}
else
{
/* FIR mode */
printk
(
"Set FIR Mode! Speed: %d
\n
"
,
speed
);
omap_writeb
((
1
<<
2
)
|
(
1
<<
6
)
|
1
,
UART3_MDR1
);
/* Set FIR mode
with SIP after each frame */
if
(
machine_is_omap_h2
())
omap_set_gpio_dataout
(
OMAP1610_H2_FIRSEL_GPIO
,
1
);
if
(
machine_is_omap_h3
())
set_h3_irda_mode
(
FIR_MODE
);
}
si
->
speed
=
speed
;
__ECHO_OUT
;
return
0
;
}
#ifdef CONFIG_PM
/*
* Suspend the IrDA interface.
*/
static
int
omap1610_irda_suspend
(
struct
device
*
_dev
,
u32
state
,
u32
level
)
{
struct
net_device
*
dev
=
dev_get_drvdata
(
_dev
);
struct
omap1610_irda
*
si
=
dev
->
priv
;
if
(
!
dev
||
level
!=
SUSPEND_DISABLE
)
return
0
;
if
(
si
->
open
)
{
/*
* Stop the transmit queue
*/
netif_device_detach
(
dev
);
disable_irq
(
dev
->
irq
);
omap1610_irda_shutdown
(
si
);
}
return
0
;
}
/*
* Resume the IrDA interface.
*/
static
int
omap1610_irda_resume
(
struct
device
*
_dev
,
u32
level
)
{
struct
net_device
*
dev
=
dev_get_drvdata
(
_dev
);
struct
omap1610_irda
*
si
=
dev
->
priv
;
if
(
!
dev
||
level
!=
RESUME_ENABLE
)
return
0
;
if
(
si
->
open
)
{
/*
* If we missed a speed change, initialise at the new speed
* directly. It is debatable whether this is actually
* required, but in the interests of continuing from where
* we left off it is desireable. The converse argument is
* that we should re-negotiate at 9600 baud again.
*/
if
(
si
->
newspeed
)
{
si
->
speed
=
si
->
newspeed
;
si
->
newspeed
=
0
;
}
omap1610_irda_startup
(
dev
);
omap1610_irda_set_speed
(
dev
,
si
->
speed
);
enable_irq
(
dev
->
irq
);
/*
* This automatically wakes up the queue
*/
netif_device_attach
(
dev
);
}
return
0
;
}
#else
#define omap1610_irda_suspend NULL
#define omap1610_irda_resume NULL
#endif
static
int
omap1610_irda_probe
(
struct
device
*
_dev
)
{
struct
platform_device
*
pdev
=
to_platform_device
(
_dev
);
struct
net_device
*
dev
;
struct
omap1610_irda
*
si
;
unsigned
int
baudrate_mask
;
int
err
=
0
;
dev
=
alloc_irdadev
(
sizeof
(
struct
omap1610_irda
));
if
(
!
dev
)
goto
err_mem_1
;
si
=
dev
->
priv
;
si
->
dev
=
&
pdev
->
dev
;
dev
->
hard_start_xmit
=
omap1610_irda_hard_xmit
;
dev
->
open
=
omap1610_irda_start
;
dev
->
stop
=
omap1610_irda_stop
;
dev
->
do_ioctl
=
omap1610_irda_ioctl
;
dev
->
get_stats
=
omap1610_irda_stats
;
dev
->
irq
=
INT_UART3
;
irda_init_max_qos_capabilies
(
&
si
->
qos
);
/*
* OMAP1610 supports SIR, MIR, FIR modes,
* but actualy supported modes depend on hardware implementation.
* OMAP1610 Innovator supports only SIR and
* OMAP1610 H2 supports both SIR and FIR
*/
baudrate_mask
=
IR_9600
|
IR_19200
|
IR_38400
|
IR_57600
|
IR_115200
|
IR_576000
|
IR_1152000
;
if
(
machine_is_omap_h2
()
||
machine_is_omap_h3
())
{
baudrate_mask
|=
(
IR_4000000
<<
8
);
}
si
->
qos
.
baud_rate
.
bits
&=
baudrate_mask
;
si
->
qos
.
min_turn_time
.
bits
=
7
;
irda_qos_bits_to_value
(
&
si
->
qos
);
err
=
register_netdev
(
dev
);
if
(
!
err
)
dev_set_drvdata
(
&
pdev
->
dev
,
dev
);
else
free_netdev
(
dev
);
err_mem_1:
return
err
;
}
static
int
omap1610_irda_remove
(
struct
device
*
_dev
)
{
struct
net_device
*
dev
=
dev_get_drvdata
(
_dev
);
#ifdef CONFIG_MACH_OMAP_H3
if
(
machine_is_omap_h3
())
cancel_delayed_work
(
&
set_h3_gpio_expa_work
);
#endif
if
(
dev
)
{
unregister_netdev
(
dev
);
free_netdev
(
dev
);
}
return
0
;
}
static
void
irda_dummy_release
(
struct
device
*
dev
)
{
/* Dummy function to keep the platform driver happy */
}
static
struct
device_driver
omap1610ir_driver
=
{
.
name
=
"omap1610-ir"
,
.
bus
=
&
platform_bus_type
,
.
probe
=
omap1610_irda_probe
,
.
remove
=
omap1610_irda_remove
,
.
suspend
=
omap1610_irda_suspend
,
.
resume
=
omap1610_irda_resume
,
};
static
struct
platform_device
omap1610ir_device
=
{
.
name
=
"omap1610-ir"
,
.
dev
.
release
=
irda_dummy_release
,
.
id
=
0
,
};
static
int
__init
omap1610_irda_init
(
void
)
{
int
ret
;
ret
=
driver_register
(
&
omap1610ir_driver
);
if
(
ret
==
0
)
{
ret
=
platform_device_register
(
&
omap1610ir_device
);
if
(
ret
)
driver_unregister
(
&
omap1610ir_driver
);
}
return
ret
;
}
static
void
__exit
omap1610_irda_exit
(
void
)
{
driver_unregister
(
&
omap1610ir_driver
);
platform_device_unregister
(
&
omap1610ir_device
);
}
module_init
(
omap1610_irda_init
);
module_exit
(
omap1610_irda_exit
);
MODULE_AUTHOR
(
"MontaVista"
);
MODULE_DESCRIPTION
(
"OMAP IrDA Driver"
);
MODULE_LICENSE
(
"GPL"
);
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment