#include <linux/io.h>
#include <linux/kernel.h>
#include <linux/list.h>
+#include <linux/mfd/syscon.h>
+#include <linux/mfd/syscon/imx6q-iomuxc-gpr.h>
#include <linux/module.h>
#include <linux/of.h>
#include <linux/of_device.h>
#include <linux/platform_device.h>
#include <linux/regulator/consumer.h>
+#include <linux/regmap.h>
#define DRV_NAME "flexcan"
(FLEXCAN_ESR_ERR_BUS | FLEXCAN_ESR_ERR_STATE)
#define FLEXCAN_ESR_ALL_INT \
(FLEXCAN_ESR_TWRN_INT | FLEXCAN_ESR_RWRN_INT | \
- FLEXCAN_ESR_BOFF_INT | FLEXCAN_ESR_ERR_INT)
+ FLEXCAN_ESR_BOFF_INT | FLEXCAN_ESR_ERR_INT | \
+ FLEXCAN_ESR_WAK_INT)
/* FLEXCAN interrupt flag register (IFLAG) bits */
#define FLEXCAN_TX_BUF_ID 8
struct flexcan_platform_data *pdata;
const struct flexcan_devtype_data *devtype_data;
struct regulator *reg_xceiver;
+ struct regmap *gpr;
+ int id;
};
static struct flexcan_devtype_data fsl_p1010_devtype_data = {
}
#endif
+static inline void flexcan_enter_stop_mode(struct flexcan_priv *priv)
+{
+ int val;
+
+ /* enable stop request */
+ if (priv->devtype_data->features & FLEXCAN_HAS_V10_FEATURES) {
+ val = priv->id ? IMX6Q_GPR13_CAN2_STOP_REQ :
+ IMX6Q_GPR13_CAN1_STOP_REQ;
+ regmap_update_bits(priv->gpr, IOMUXC_GPR13,
+ val, val);
+ }
+}
+
+static inline void flexcan_exit_stop_mode(struct flexcan_priv *priv)
+{
+ int val;
+
+ /* remove stop request */
+ if (priv->devtype_data->features & FLEXCAN_HAS_V10_FEATURES) {
+ val = priv->id ? IMX6Q_GPR13_CAN2_STOP_REQ :
+ IMX6Q_GPR13_CAN1_STOP_REQ;
+ regmap_update_bits(priv->gpr, IOMUXC_GPR13,
+ val, 0);
+ }
+}
+
static inline int flexcan_transceiver_enable(const struct flexcan_priv *priv)
{
if (priv->pdata && priv->pdata->transceiver_switch) {
if (reg_esr & FLEXCAN_ESR_ALL_INT)
flexcan_write(reg_esr & FLEXCAN_ESR_ALL_INT, ®s->esr);
+ if (reg_esr & FLEXCAN_ESR_WAK_INT)
+ flexcan_exit_stop_mode(priv);
+
/*
* schedule NAPI in case of:
* - rx IRQ
* enable warning int
* choose format C
* disable local echo
- *
+ * enable self wakeup
*/
reg_mcr = flexcan_read(®s->mcr);
reg_mcr &= ~FLEXCAN_MCR_MAXMB(0xff);
reg_mcr |= FLEXCAN_MCR_FRZ | FLEXCAN_MCR_FEN | FLEXCAN_MCR_HALT |
FLEXCAN_MCR_SUPV | FLEXCAN_MCR_WRN_EN |
FLEXCAN_MCR_IDAM_C | FLEXCAN_MCR_SRX_DIS |
+ FLEXCAN_MCR_WAK_MSK | FLEXCAN_MCR_SLF_WAK |
FLEXCAN_MCR_MAXMB(FLEXCAN_TX_BUF_ID);
netdev_dbg(dev, "%s: writing mcr=0x%08x", __func__, reg_mcr);
flexcan_write(reg_mcr, ®s->mcr);
void __iomem *base;
int err, irq;
u32 clock_freq = 0;
+ int wakeup = 1;
if (pdev->dev.of_node)
of_property_read_u32(pdev->dev.of_node,
devm_can_led_init(dev);
+ if (priv->devtype_data->features & FLEXCAN_HAS_V10_FEATURES) {
+ priv->gpr = syscon_regmap_lookup_by_phandle(pdev->dev.of_node,
+ "gpr");
+ if (IS_ERR(priv->gpr)) {
+ wakeup = 0;
+ dev_dbg(&pdev->dev, "can not get grp\n");
+ }
+
+ priv->id = of_alias_get_id(pdev->dev.of_node, "flexcan");
+ if (priv->id < 0) {
+ wakeup = 0;
+ dev_dbg(&pdev->dev, "can not get alias id\n");
+ }
+ }
+
+ device_set_wakeup_capable(&pdev->dev, wakeup);
+
dev_info(&pdev->dev, "device registered (reg_base=%p, irq=%d)\n",
priv->base, dev->irq);
if (netif_running(dev)) {
netif_stop_queue(dev);
netif_device_detach(dev);
+ /*
+ * if wakeup is enabled, enter stop mode
+ * else enter disabled mode.
+ */
+ if (device_may_wakeup(device)) {
+ enable_irq_wake(dev->irq);
+ flexcan_enter_stop_mode(priv);
+ } else {
+ flexcan_chip_disable(priv);
+ }
+ } else {
+ flexcan_chip_disable(priv);
}
priv->can.state = CAN_STATE_SLEEPING;
if (netif_running(dev)) {
netif_device_attach(dev);
netif_start_queue(dev);
+
+ if (device_may_wakeup(device)) {
+ disable_irq_wake(dev->irq);
+ flexcan_exit_stop_mode(priv);
+ } else {
+ flexcan_chip_enable(priv);
+ }
+ } else {
+ flexcan_chip_enable(priv);
}
- return flexcan_chip_enable(priv);
+
+ return 0;
}
#endif /* CONFIG_PM_SLEEP */