2020-02-22 22:08:49 -06:00
// SPDX-License-Identifier: GPL-2.0
//
// Copyright (c) 2017-2019 Samuel Holland <samuel@sholland.org>
# include <linux/bitops.h>
# include <linux/clk.h>
# include <linux/device.h>
# include <linux/err.h>
# include <linux/interrupt.h>
# include <linux/io.h>
# include <linux/kernel.h>
# include <linux/mailbox_controller.h>
# include <linux/module.h>
# include <linux/of.h>
# include <linux/of_irq.h>
# include <linux/platform_device.h>
# include <linux/reset.h>
# include <linux/spinlock.h>
# define NUM_CHANS 8
# define CTRL_REG(n) (0x0000 + 0x4 * ((n) / 4))
# define CTRL_RX(n) BIT(0 + 8 * ((n) % 4))
# define CTRL_TX(n) BIT(4 + 8 * ((n) % 4))
# define REMOTE_IRQ_EN_REG 0x0040
# define REMOTE_IRQ_STAT_REG 0x0050
# define LOCAL_IRQ_EN_REG 0x0060
# define LOCAL_IRQ_STAT_REG 0x0070
# define RX_IRQ(n) BIT(0 + 2 * (n))
# define RX_IRQ_MASK 0x5555
# define TX_IRQ(n) BIT(1 + 2 * (n))
# define TX_IRQ_MASK 0xaaaa
# define FIFO_STAT_REG(n) (0x0100 + 0x4 * (n))
# define FIFO_STAT_MASK GENMASK(0, 0)
# define MSG_STAT_REG(n) (0x0140 + 0x4 * (n))
# define MSG_STAT_MASK GENMASK(2, 0)
# define MSG_DATA_REG(n) (0x0180 + 0x4 * (n))
# define mbox_dbg(mbox, ...) dev_dbg((mbox)->controller.dev, __VA_ARGS__)
struct sun6i_msgbox {
struct mbox_controller controller ;
struct clk * clk ;
spinlock_t lock ;
void __iomem * regs ;
} ;
static bool sun6i_msgbox_last_tx_done ( struct mbox_chan * chan ) ;
static bool sun6i_msgbox_peek_data ( struct mbox_chan * chan ) ;
static inline int channel_number ( struct mbox_chan * chan )
{
return chan - chan - > mbox - > chans ;
}
static inline struct sun6i_msgbox * to_sun6i_msgbox ( struct mbox_chan * chan )
{
return chan - > con_priv ;
}
static irqreturn_t sun6i_msgbox_irq ( int irq , void * dev_id )
{
struct sun6i_msgbox * mbox = dev_id ;
uint32_t status ;
int n ;
/* Only examine channels that are currently enabled. */
status = readl ( mbox - > regs + LOCAL_IRQ_EN_REG ) &
readl ( mbox - > regs + LOCAL_IRQ_STAT_REG ) ;
if ( ! ( status & RX_IRQ_MASK ) )
return IRQ_NONE ;
for ( n = 0 ; n < NUM_CHANS ; + + n ) {
struct mbox_chan * chan = & mbox - > controller . chans [ n ] ;
if ( ! ( status & RX_IRQ ( n ) ) )
continue ;
while ( sun6i_msgbox_peek_data ( chan ) ) {
uint32_t msg = readl ( mbox - > regs + MSG_DATA_REG ( n ) ) ;
mbox_dbg ( mbox , " Channel %d received 0x%08x \n " , n , msg ) ;
mbox_chan_received_data ( chan , & msg ) ;
}
/* The IRQ can be cleared only once the FIFO is empty. */
writel ( RX_IRQ ( n ) , mbox - > regs + LOCAL_IRQ_STAT_REG ) ;
}
return IRQ_HANDLED ;
}
static int sun6i_msgbox_send_data ( struct mbox_chan * chan , void * data )
{
struct sun6i_msgbox * mbox = to_sun6i_msgbox ( chan ) ;
int n = channel_number ( chan ) ;
uint32_t msg = * ( uint32_t * ) data ;
/* Using a channel backwards gets the hardware into a bad state. */
if ( WARN_ON_ONCE ( ! ( readl ( mbox - > regs + CTRL_REG ( n ) ) & CTRL_TX ( n ) ) ) )
return 0 ;
writel ( msg , mbox - > regs + MSG_DATA_REG ( n ) ) ;
mbox_dbg ( mbox , " Channel %d sent 0x%08x \n " , n , msg ) ;
return 0 ;
}
static int sun6i_msgbox_startup ( struct mbox_chan * chan )
{
struct sun6i_msgbox * mbox = to_sun6i_msgbox ( chan ) ;
int n = channel_number ( chan ) ;
/* The coprocessor is responsible for setting channel directions. */
if ( readl ( mbox - > regs + CTRL_REG ( n ) ) & CTRL_RX ( n ) ) {
/* Flush the receive FIFO. */
while ( sun6i_msgbox_peek_data ( chan ) )
readl ( mbox - > regs + MSG_DATA_REG ( n ) ) ;
writel ( RX_IRQ ( n ) , mbox - > regs + LOCAL_IRQ_STAT_REG ) ;
/* Enable the receive IRQ. */
spin_lock ( & mbox - > lock ) ;
writel ( readl ( mbox - > regs + LOCAL_IRQ_EN_REG ) | RX_IRQ ( n ) ,
mbox - > regs + LOCAL_IRQ_EN_REG ) ;
spin_unlock ( & mbox - > lock ) ;
}
mbox_dbg ( mbox , " Channel %d startup complete \n " , n ) ;
return 0 ;
}
static void sun6i_msgbox_shutdown ( struct mbox_chan * chan )
{
struct sun6i_msgbox * mbox = to_sun6i_msgbox ( chan ) ;
int n = channel_number ( chan ) ;
if ( readl ( mbox - > regs + CTRL_REG ( n ) ) & CTRL_RX ( n ) ) {
/* Disable the receive IRQ. */
spin_lock ( & mbox - > lock ) ;
writel ( readl ( mbox - > regs + LOCAL_IRQ_EN_REG ) & ~ RX_IRQ ( n ) ,
mbox - > regs + LOCAL_IRQ_EN_REG ) ;
spin_unlock ( & mbox - > lock ) ;
/* Attempt to flush the FIFO until the IRQ is cleared. */
do {
while ( sun6i_msgbox_peek_data ( chan ) )
readl ( mbox - > regs + MSG_DATA_REG ( n ) ) ;
writel ( RX_IRQ ( n ) , mbox - > regs + LOCAL_IRQ_STAT_REG ) ;
} while ( readl ( mbox - > regs + LOCAL_IRQ_STAT_REG ) & RX_IRQ ( n ) ) ;
}
mbox_dbg ( mbox , " Channel %d shutdown complete \n " , n ) ;
}
static bool sun6i_msgbox_last_tx_done ( struct mbox_chan * chan )
{
struct sun6i_msgbox * mbox = to_sun6i_msgbox ( chan ) ;
int n = channel_number ( chan ) ;
/*
* The hardware allows snooping on the remote user ' s IRQ statuses .
* We consider a message to be acknowledged only once the receive IRQ
* for that channel is cleared . Since the receive IRQ for a channel
* cannot be cleared until the FIFO for that channel is empty , this
* ensures that the message has actually been read . It also gives the
* recipient an opportunity to perform minimal processing before
* acknowledging the message .
*/
return ! ( readl ( mbox - > regs + REMOTE_IRQ_STAT_REG ) & RX_IRQ ( n ) ) ;
}
static bool sun6i_msgbox_peek_data ( struct mbox_chan * chan )
{
struct sun6i_msgbox * mbox = to_sun6i_msgbox ( chan ) ;
int n = channel_number ( chan ) ;
return readl ( mbox - > regs + MSG_STAT_REG ( n ) ) & MSG_STAT_MASK ;
}
static const struct mbox_chan_ops sun6i_msgbox_chan_ops = {
. send_data = sun6i_msgbox_send_data ,
. startup = sun6i_msgbox_startup ,
. shutdown = sun6i_msgbox_shutdown ,
. last_tx_done = sun6i_msgbox_last_tx_done ,
. peek_data = sun6i_msgbox_peek_data ,
} ;
static int sun6i_msgbox_probe ( struct platform_device * pdev )
{
struct device * dev = & pdev - > dev ;
struct mbox_chan * chans ;
struct reset_control * reset ;
struct sun6i_msgbox * mbox ;
int i , ret ;
mbox = devm_kzalloc ( dev , sizeof ( * mbox ) , GFP_KERNEL ) ;
if ( ! mbox )
return - ENOMEM ;
chans = devm_kcalloc ( dev , NUM_CHANS , sizeof ( * chans ) , GFP_KERNEL ) ;
if ( ! chans )
return - ENOMEM ;
for ( i = 0 ; i < NUM_CHANS ; + + i )
chans [ i ] . con_priv = mbox ;
mbox - > clk = devm_clk_get ( dev , NULL ) ;
if ( IS_ERR ( mbox - > clk ) ) {
ret = PTR_ERR ( mbox - > clk ) ;
dev_err ( dev , " Failed to get clock: %d \n " , ret ) ;
return ret ;
}
ret = clk_prepare_enable ( mbox - > clk ) ;
if ( ret ) {
dev_err ( dev , " Failed to enable clock: %d \n " , ret ) ;
return ret ;
}
reset = devm_reset_control_get_exclusive ( dev , NULL ) ;
if ( IS_ERR ( reset ) ) {
ret = PTR_ERR ( reset ) ;
dev_err ( dev , " Failed to get reset control: %d \n " , ret ) ;
goto err_disable_unprepare ;
}
/*
* NOTE : We rely on platform firmware to preconfigure the channel
* directions , and we share this hardware block with other firmware
* that runs concurrently with Linux ( e . g . a trusted monitor ) .
*
* Therefore , we do * not * assert the reset line if probing fails or
* when removing the device .
*/
ret = reset_control_deassert ( reset ) ;
if ( ret ) {
dev_err ( dev , " Failed to deassert reset: %d \n " , ret ) ;
goto err_disable_unprepare ;
}
2021-09-07 15:42:00 +08:00
mbox - > regs = devm_platform_ioremap_resource ( pdev , 0 ) ;
2020-02-22 22:08:49 -06:00
if ( IS_ERR ( mbox - > regs ) ) {
ret = PTR_ERR ( mbox - > regs ) ;
dev_err ( dev , " Failed to map MMIO resource: %d \n " , ret ) ;
goto err_disable_unprepare ;
}
/* Disable all IRQs for this end of the msgbox. */
writel ( 0 , mbox - > regs + LOCAL_IRQ_EN_REG ) ;
ret = devm_request_irq ( dev , irq_of_parse_and_map ( dev - > of_node , 0 ) ,
sun6i_msgbox_irq , 0 , dev_name ( dev ) , mbox ) ;
if ( ret ) {
dev_err ( dev , " Failed to register IRQ handler: %d \n " , ret ) ;
goto err_disable_unprepare ;
}
mbox - > controller . dev = dev ;
mbox - > controller . ops = & sun6i_msgbox_chan_ops ;
mbox - > controller . chans = chans ;
mbox - > controller . num_chans = NUM_CHANS ;
mbox - > controller . txdone_irq = false ;
mbox - > controller . txdone_poll = true ;
mbox - > controller . txpoll_period = 5 ;
spin_lock_init ( & mbox - > lock ) ;
platform_set_drvdata ( pdev , mbox ) ;
ret = mbox_controller_register ( & mbox - > controller ) ;
if ( ret ) {
dev_err ( dev , " Failed to register controller: %d \n " , ret ) ;
goto err_disable_unprepare ;
}
return 0 ;
err_disable_unprepare :
clk_disable_unprepare ( mbox - > clk ) ;
return ret ;
}
static int sun6i_msgbox_remove ( struct platform_device * pdev )
{
struct sun6i_msgbox * mbox = platform_get_drvdata ( pdev ) ;
mbox_controller_unregister ( & mbox - > controller ) ;
/* See the comment in sun6i_msgbox_probe about the reset line. */
clk_disable_unprepare ( mbox - > clk ) ;
return 0 ;
}
static const struct of_device_id sun6i_msgbox_of_match [ ] = {
{ . compatible = " allwinner,sun6i-a31-msgbox " , } ,
{ } ,
} ;
MODULE_DEVICE_TABLE ( of , sun6i_msgbox_of_match ) ;
static struct platform_driver sun6i_msgbox_driver = {
. driver = {
. name = " sun6i-msgbox " ,
. of_match_table = sun6i_msgbox_of_match ,
} ,
. probe = sun6i_msgbox_probe ,
. remove = sun6i_msgbox_remove ,
} ;
module_platform_driver ( sun6i_msgbox_driver ) ;
MODULE_AUTHOR ( " Samuel Holland <samuel@sholland.org> " ) ;
MODULE_DESCRIPTION ( " Allwinner sun6i/sun8i/sun9i/sun50i Message Box " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;