ASoC: sti: helper functions for unip tdm slots configuration
- sti_uniperiph_set_tdm_slot: store tdm slot config in unip context - sti_uniperiph_get_tdm_word_pos: configure unip tdm slots pos regs Signed-off-by: Moise Gergaud <moise.gergaud@st.com> Acked-by: Arnaud Pouliquen <arnaud.pouliquen@st.com> Signed-off-by: Mark Brown <broonie@kernel.org>
This commit is contained in:
parent
9a00a3e9fe
commit
44f948bdb1
@ -10,6 +10,96 @@
|
|||||||
|
|
||||||
#include "uniperif.h"
|
#include "uniperif.h"
|
||||||
|
|
||||||
|
/*
|
||||||
|
* User frame size shall be 2, 4, 6 or 8 32-bits words length
|
||||||
|
* (i.e. 8, 16, 24 or 32 bytes)
|
||||||
|
* This constraint comes from allowed values for
|
||||||
|
* UNIPERIF_I2S_FMT_NUM_CH register
|
||||||
|
*/
|
||||||
|
#define UNIPERIF_MAX_FRAME_SZ 0x20
|
||||||
|
#define UNIPERIF_ALLOWED_FRAME_SZ (0x08 | 0x10 | 0x18 | UNIPERIF_MAX_FRAME_SZ)
|
||||||
|
|
||||||
|
int sti_uniperiph_set_tdm_slot(struct snd_soc_dai *dai, unsigned int tx_mask,
|
||||||
|
unsigned int rx_mask, int slots,
|
||||||
|
int slot_width)
|
||||||
|
{
|
||||||
|
struct sti_uniperiph_data *priv = snd_soc_dai_get_drvdata(dai);
|
||||||
|
struct uniperif *uni = priv->dai_data.uni;
|
||||||
|
int i, frame_size, avail_slots;
|
||||||
|
|
||||||
|
if (!UNIPERIF_TYPE_IS_TDM(uni)) {
|
||||||
|
dev_err(uni->dev, "cpu dai not in tdm mode\n");
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* store info in unip context */
|
||||||
|
uni->tdm_slot.slots = slots;
|
||||||
|
uni->tdm_slot.slot_width = slot_width;
|
||||||
|
/* unip is unidirectionnal */
|
||||||
|
uni->tdm_slot.mask = (tx_mask != 0) ? tx_mask : rx_mask;
|
||||||
|
|
||||||
|
/* number of available timeslots */
|
||||||
|
for (i = 0, avail_slots = 0; i < uni->tdm_slot.slots; i++) {
|
||||||
|
if ((uni->tdm_slot.mask >> i) & 0x01)
|
||||||
|
avail_slots++;
|
||||||
|
}
|
||||||
|
uni->tdm_slot.avail_slots = avail_slots;
|
||||||
|
|
||||||
|
/* frame size in bytes */
|
||||||
|
frame_size = uni->tdm_slot.avail_slots * uni->tdm_slot.slot_width / 8;
|
||||||
|
|
||||||
|
/* check frame size is allowed */
|
||||||
|
if ((frame_size > UNIPERIF_MAX_FRAME_SZ) ||
|
||||||
|
(frame_size & ~(int)UNIPERIF_ALLOWED_FRAME_SZ)) {
|
||||||
|
dev_err(uni->dev, "frame size not allowed: %d bytes\n",
|
||||||
|
frame_size);
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int sti_uniperiph_get_tdm_word_pos(struct uniperif *uni,
|
||||||
|
unsigned int *word_pos)
|
||||||
|
{
|
||||||
|
int slot_width = uni->tdm_slot.slot_width / 8;
|
||||||
|
int slots_num = uni->tdm_slot.slots;
|
||||||
|
unsigned int slots_mask = uni->tdm_slot.mask;
|
||||||
|
int i, j, k;
|
||||||
|
unsigned int word16_pos[4];
|
||||||
|
|
||||||
|
/* word16_pos:
|
||||||
|
* word16_pos[0] = WORDX_LSB
|
||||||
|
* word16_pos[1] = WORDX_MSB,
|
||||||
|
* word16_pos[2] = WORDX+1_LSB
|
||||||
|
* word16_pos[3] = WORDX+1_MSB
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* set unip word position */
|
||||||
|
for (i = 0, j = 0, k = 0; (i < slots_num) && (k < WORD_MAX); i++) {
|
||||||
|
if ((slots_mask >> i) & 0x01) {
|
||||||
|
word16_pos[j] = i * slot_width;
|
||||||
|
|
||||||
|
if (slot_width == 4) {
|
||||||
|
word16_pos[j + 1] = word16_pos[j] + 2;
|
||||||
|
j++;
|
||||||
|
}
|
||||||
|
j++;
|
||||||
|
|
||||||
|
if (j > 3) {
|
||||||
|
word_pos[k] = word16_pos[1] |
|
||||||
|
(word16_pos[0] << 8) |
|
||||||
|
(word16_pos[3] << 16) |
|
||||||
|
(word16_pos[2] << 24);
|
||||||
|
j = 0;
|
||||||
|
k++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* sti_uniperiph_dai_create_ctrl
|
* sti_uniperiph_dai_create_ctrl
|
||||||
* This function is used to create Ctrl associated to DAI but also pcm device.
|
* This function is used to create Ctrl associated to DAI but also pcm device.
|
||||||
|
@ -1270,6 +1270,14 @@ enum uniperif_iec958_encoding_mode {
|
|||||||
UNIPERIF_IEC958_ENCODING_MODE_ENCODED
|
UNIPERIF_IEC958_ENCODING_MODE_ENCODED
|
||||||
};
|
};
|
||||||
|
|
||||||
|
enum uniperif_word_pos {
|
||||||
|
WORD_1_2,
|
||||||
|
WORD_3_4,
|
||||||
|
WORD_5_6,
|
||||||
|
WORD_7_8,
|
||||||
|
WORD_MAX
|
||||||
|
};
|
||||||
|
|
||||||
struct uniperif_info {
|
struct uniperif_info {
|
||||||
int id; /* instance value of the uniperipheral IP */
|
int id; /* instance value of the uniperipheral IP */
|
||||||
enum uniperif_type type;
|
enum uniperif_type type;
|
||||||
@ -1281,6 +1289,13 @@ struct uniperif_iec958_settings {
|
|||||||
struct snd_aes_iec958 iec958;
|
struct snd_aes_iec958 iec958;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct dai_tdm_slot {
|
||||||
|
unsigned int mask;
|
||||||
|
int slots;
|
||||||
|
int slot_width;
|
||||||
|
unsigned int avail_slots;
|
||||||
|
};
|
||||||
|
|
||||||
struct uniperif {
|
struct uniperif {
|
||||||
/* System information */
|
/* System information */
|
||||||
struct uniperif_info *info;
|
struct uniperif_info *info;
|
||||||
@ -1317,6 +1332,7 @@ struct uniperif {
|
|||||||
|
|
||||||
/* dai properties */
|
/* dai properties */
|
||||||
unsigned int daifmt;
|
unsigned int daifmt;
|
||||||
|
struct dai_tdm_slot tdm_slot;
|
||||||
|
|
||||||
/* DAI callbacks */
|
/* DAI callbacks */
|
||||||
const struct snd_soc_dai_ops *dai_ops;
|
const struct snd_soc_dai_ops *dai_ops;
|
||||||
@ -1373,4 +1389,11 @@ int sti_uniperiph_dai_hw_params(struct snd_pcm_substream *substream,
|
|||||||
struct snd_pcm_hw_params *params,
|
struct snd_pcm_hw_params *params,
|
||||||
struct snd_soc_dai *dai);
|
struct snd_soc_dai *dai);
|
||||||
|
|
||||||
|
int sti_uniperiph_set_tdm_slot(struct snd_soc_dai *dai, unsigned int tx_mask,
|
||||||
|
unsigned int rx_mask, int slots,
|
||||||
|
int slot_width);
|
||||||
|
|
||||||
|
int sti_uniperiph_get_tdm_word_pos(struct uniperif *uni,
|
||||||
|
unsigned int *word_pos);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
Loading…
x
Reference in New Issue
Block a user