forked from Minki/linux
4e552310cd
msg->complete() should always be called from process context once irq_safe runtime pm flag is no longer set for omap-ssi. Signed-off-by: Sebastian Reichel <sre@kernel.org> Tested-by: Pavel Machek <pavel@ucw.cz>
178 lines
5.3 KiB
C
178 lines
5.3 KiB
C
/* OMAP SSI internal interface.
|
|
*
|
|
* Copyright (C) 2010 Nokia Corporation. All rights reserved.
|
|
* Copyright (C) 2013 Sebastian Reichel
|
|
*
|
|
* Contact: Carlos Chinea <carlos.chinea@nokia.com>
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License
|
|
* version 2 as published by the Free Software Foundation.
|
|
*
|
|
* This program is distributed in the hope that it will be useful, but
|
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* General Public License for more details.
|
|
*
|
|
* 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., 51 Franklin St, Fifth Floor, Boston, MA
|
|
* 02110-1301 USA
|
|
*/
|
|
|
|
#ifndef __LINUX_HSI_OMAP_SSI_H__
|
|
#define __LINUX_HSI_OMAP_SSI_H__
|
|
|
|
#include <linux/device.h>
|
|
#include <linux/module.h>
|
|
#include <linux/platform_device.h>
|
|
#include <linux/hsi/hsi.h>
|
|
#include <linux/gpio/consumer.h>
|
|
#include <linux/interrupt.h>
|
|
#include <linux/io.h>
|
|
|
|
#define SSI_MAX_CHANNELS 8
|
|
#define SSI_MAX_GDD_LCH 8
|
|
#define SSI_BYTES_TO_FRAMES(x) ((((x) - 1) >> 2) + 1)
|
|
|
|
#define SSI_WAKE_EN 0
|
|
|
|
/**
|
|
* struct omap_ssm_ctx - OMAP synchronous serial module (TX/RX) context
|
|
* @mode: Bit transmission mode
|
|
* @channels: Number of channels
|
|
* @framesize: Frame size in bits
|
|
* @timeout: RX frame timeout
|
|
* @divisor: TX divider
|
|
* @arb_mode: Arbitration mode for TX frame (Round robin, priority)
|
|
*/
|
|
struct omap_ssm_ctx {
|
|
u32 mode;
|
|
u32 channels;
|
|
u32 frame_size;
|
|
union {
|
|
u32 timeout; /* Rx Only */
|
|
struct {
|
|
u32 arb_mode;
|
|
u32 divisor;
|
|
}; /* Tx only */
|
|
};
|
|
};
|
|
|
|
/**
|
|
* struct omap_ssi_port - OMAP SSI port data
|
|
* @dev: device associated to the port (HSI port)
|
|
* @pdev: platform device associated to the port
|
|
* @sst_dma: SSI transmitter physical base address
|
|
* @ssr_dma: SSI receiver physical base address
|
|
* @sst_base: SSI transmitter base address
|
|
* @ssr_base: SSI receiver base address
|
|
* @wk_lock: spin lock to serialize access to the wake lines
|
|
* @lock: Spin lock to serialize access to the SSI port
|
|
* @channels: Current number of channels configured (1,2,4 or 8)
|
|
* @txqueue: TX message queues
|
|
* @rxqueue: RX message queues
|
|
* @brkqueue: Queue of incoming HWBREAK requests (FRAME mode)
|
|
* @errqueue: Queue for failed messages
|
|
* @errqueue_work: Delayed Work for failed messages
|
|
* @irq: IRQ number
|
|
* @wake_irq: IRQ number for incoming wake line (-1 if none)
|
|
* @wake_gpio: GPIO number for incoming wake line (-1 if none)
|
|
* @flags: flags to keep track of states
|
|
* @wk_refcount: Reference count for output wake line
|
|
* @work: worker for starting TX
|
|
* @sys_mpu_enable: Context for the interrupt enable register for irq 0
|
|
* @sst: Context for the synchronous serial transmitter
|
|
* @ssr: Context for the synchronous serial receiver
|
|
*/
|
|
struct omap_ssi_port {
|
|
struct device *dev;
|
|
struct device *pdev;
|
|
dma_addr_t sst_dma;
|
|
dma_addr_t ssr_dma;
|
|
void __iomem *sst_base;
|
|
void __iomem *ssr_base;
|
|
spinlock_t wk_lock;
|
|
spinlock_t lock;
|
|
unsigned int channels;
|
|
struct list_head txqueue[SSI_MAX_CHANNELS];
|
|
struct list_head rxqueue[SSI_MAX_CHANNELS];
|
|
struct list_head brkqueue;
|
|
struct list_head errqueue;
|
|
struct delayed_work errqueue_work;
|
|
unsigned int irq;
|
|
int wake_irq;
|
|
struct gpio_desc *wake_gpio;
|
|
bool wktest:1; /* FIXME: HACK to be removed */
|
|
unsigned long flags;
|
|
unsigned int wk_refcount;
|
|
struct work_struct work;
|
|
/* OMAP SSI port context */
|
|
u32 sys_mpu_enable; /* We use only one irq */
|
|
struct omap_ssm_ctx sst;
|
|
struct omap_ssm_ctx ssr;
|
|
u32 loss_count;
|
|
u32 port_id;
|
|
#ifdef CONFIG_DEBUG_FS
|
|
struct dentry *dir;
|
|
#endif
|
|
};
|
|
|
|
/**
|
|
* struct gdd_trn - GDD transaction data
|
|
* @msg: Pointer to the HSI message being served
|
|
* @sg: Pointer to the current sg entry being served
|
|
*/
|
|
struct gdd_trn {
|
|
struct hsi_msg *msg;
|
|
struct scatterlist *sg;
|
|
};
|
|
|
|
/**
|
|
* struct omap_ssi_controller - OMAP SSI controller data
|
|
* @dev: device associated to the controller (HSI controller)
|
|
* @sys: SSI I/O base address
|
|
* @gdd: GDD I/O base address
|
|
* @fck: SSI functional clock
|
|
* @gdd_irq: IRQ line for GDD
|
|
* @gdd_tasklet: bottom half for DMA transfers
|
|
* @gdd_trn: Array of GDD transaction data for ongoing GDD transfers
|
|
* @lock: lock to serialize access to GDD
|
|
* @fck_nb: DVFS notfifier block
|
|
* @fck_rate: clock rate
|
|
* @loss_count: To follow if we need to restore context or not
|
|
* @max_speed: Maximum TX speed (Kb/s) set by the clients.
|
|
* @gdd_gcr: SSI GDD saved context
|
|
* @get_loss: Pointer to omap_pm_get_dev_context_loss_count, if any
|
|
* @port: Array of pointers of the ports of the controller
|
|
* @dir: Debugfs SSI root directory
|
|
*/
|
|
struct omap_ssi_controller {
|
|
struct device *dev;
|
|
void __iomem *sys;
|
|
void __iomem *gdd;
|
|
struct clk *fck;
|
|
unsigned int gdd_irq;
|
|
struct tasklet_struct gdd_tasklet;
|
|
struct gdd_trn gdd_trn[SSI_MAX_GDD_LCH];
|
|
spinlock_t lock;
|
|
struct notifier_block fck_nb;
|
|
unsigned long fck_rate;
|
|
u32 loss_count;
|
|
u32 max_speed;
|
|
/* OMAP SSI Controller context */
|
|
u32 gdd_gcr;
|
|
int (*get_loss)(struct device *dev);
|
|
struct omap_ssi_port **port;
|
|
#ifdef CONFIG_DEBUG_FS
|
|
struct dentry *dir;
|
|
#endif
|
|
};
|
|
|
|
void omap_ssi_port_update_fclk(struct hsi_controller *ssi,
|
|
struct omap_ssi_port *omap_port);
|
|
|
|
extern struct platform_driver ssi_port_pdriver;
|
|
|
|
#endif /* __LINUX_HSI_OMAP_SSI_H__ */
|