123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154 |
- /*
- * Renesas R-Car Gen1 SRU/SSI support
- *
- * Copyright (C) 2013 Renesas Solutions Corp.
- * Kuninori Morimoto <kuninori.morimoto.gx@renesas.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.
- */
- #include "rsnd.h"
- struct rsnd_gen_ops {
- int (*path_init)(struct rsnd_priv *priv,
- struct rsnd_dai *rdai,
- struct rsnd_dai_stream *io);
- int (*path_exit)(struct rsnd_priv *priv,
- struct rsnd_dai *rdai,
- struct rsnd_dai_stream *io);
- };
- struct rsnd_gen_reg_map {
- int index; /* -1 : not supported */
- u32 offset_id; /* offset of ssi0, ssi1, ssi2... */
- u32 offset_adr; /* offset of SSICR, SSISR, ... */
- };
- struct rsnd_gen {
- void __iomem *base[RSND_BASE_MAX];
- struct rsnd_gen_reg_map reg_map[RSND_REG_MAX];
- struct rsnd_gen_ops *ops;
- };
- #define rsnd_priv_to_gen(p) ((struct rsnd_gen *)(p)->gen)
- #define rsnd_is_gen1(s) ((s)->info->flags & RSND_GEN1)
- #define rsnd_is_gen2(s) ((s)->info->flags & RSND_GEN2)
- /*
- * Gen2
- * will be filled in the future
- */
- /*
- * Gen1
- */
- static int rsnd_gen1_probe(struct platform_device *pdev,
- struct rcar_snd_info *info,
- struct rsnd_priv *priv)
- {
- return 0;
- }
- static void rsnd_gen1_remove(struct platform_device *pdev,
- struct rsnd_priv *priv)
- {
- }
- /*
- * Gen
- */
- int rsnd_gen_path_init(struct rsnd_priv *priv,
- struct rsnd_dai *rdai,
- struct rsnd_dai_stream *io)
- {
- struct rsnd_gen *gen = rsnd_priv_to_gen(priv);
- return gen->ops->path_init(priv, rdai, io);
- }
- int rsnd_gen_path_exit(struct rsnd_priv *priv,
- struct rsnd_dai *rdai,
- struct rsnd_dai_stream *io)
- {
- struct rsnd_gen *gen = rsnd_priv_to_gen(priv);
- return gen->ops->path_exit(priv, rdai, io);
- }
- void __iomem *rsnd_gen_reg_get(struct rsnd_priv *priv,
- struct rsnd_mod *mod,
- enum rsnd_reg reg)
- {
- struct rsnd_gen *gen = rsnd_priv_to_gen(priv);
- struct device *dev = rsnd_priv_to_dev(priv);
- int index;
- u32 offset_id, offset_adr;
- if (reg >= RSND_REG_MAX) {
- dev_err(dev, "rsnd_reg reg error\n");
- return NULL;
- }
- index = gen->reg_map[reg].index;
- offset_id = gen->reg_map[reg].offset_id;
- offset_adr = gen->reg_map[reg].offset_adr;
- if (index < 0) {
- dev_err(dev, "unsupported reg access %d\n", reg);
- return NULL;
- }
- if (offset_id && mod)
- offset_id *= rsnd_mod_id(mod);
- /*
- * index/offset were set on gen1/gen2
- */
- return gen->base[index] + offset_id + offset_adr;
- }
- int rsnd_gen_probe(struct platform_device *pdev,
- struct rcar_snd_info *info,
- struct rsnd_priv *priv)
- {
- struct device *dev = rsnd_priv_to_dev(priv);
- struct rsnd_gen *gen;
- int i;
- gen = devm_kzalloc(dev, sizeof(*gen), GFP_KERNEL);
- if (!gen) {
- dev_err(dev, "GEN allocate failed\n");
- return -ENOMEM;
- }
- priv->gen = gen;
- /*
- * see
- * rsnd_reg_get()
- * rsnd_gen_probe()
- */
- for (i = 0; i < RSND_REG_MAX; i++)
- gen->reg_map[i].index = -1;
- /*
- * init each module
- */
- if (rsnd_is_gen1(priv))
- return rsnd_gen1_probe(pdev, info, priv);
- dev_err(dev, "unknown generation R-Car sound device\n");
- return -ENODEV;
- }
- void rsnd_gen_remove(struct platform_device *pdev,
- struct rsnd_priv *priv)
- {
- if (rsnd_is_gen1(priv))
- rsnd_gen1_remove(pdev, priv);
- }
|