···19192020#include "stm32_sai.h"21212222+static int stm32_sai_get_parent_clk(struct stm32_sai_data *sai);2323+2224static const struct stm32_sai_conf stm32_sai_conf_f4 = {2325 .version = STM_SAI_STM32F4,2426 .fifo_size = 8,2527 .has_spdif_pdm = false,2828+ .get_sai_ck_parent = stm32_sai_get_parent_clk,2629};27302831/*2929- * Default settings for stm32 H7 socs and next.3232+ * Default settings for STM32H7x socs and STM32MP1x.3033 * These default settings will be overridden if the soc provides3134 * support of hardware configuration registers.3535+ * - STM32H7: rely on default settings3636+ * - STM32MP1: retrieve settings from registers3237 */3338static const struct stm32_sai_conf stm32_sai_conf_h7 = {3439 .version = STM_SAI_STM32H7,3540 .fifo_size = 8,3641 .has_spdif_pdm = true,4242+ .get_sai_ck_parent = stm32_sai_get_parent_clk,4343+};4444+4545+/*4646+ * STM32MP2x:4747+ * - do not use SAI parent clock source selection4848+ * - do not use DMA burst mode4949+ */5050+static const struct stm32_sai_conf stm32_sai_conf_mp25 = {5151+ .no_dma_burst = true,3752};38533954static const struct of_device_id stm32_sai_ids[] = {4055 { .compatible = "st,stm32f4-sai", .data = (void *)&stm32_sai_conf_f4 },4156 { .compatible = "st,stm32h7-sai", .data = (void *)&stm32_sai_conf_h7 },5757+ { .compatible = "st,stm32mp25-sai", .data = (void *)&stm32_sai_conf_mp25 },4258 {}4359};4460···164148 return ret;165149}166150151151+static int stm32_sai_get_parent_clk(struct stm32_sai_data *sai)152152+{153153+ struct device *dev = &sai->pdev->dev;154154+155155+ sai->clk_x8k = devm_clk_get(dev, "x8k");156156+ if (IS_ERR(sai->clk_x8k)) {157157+ if (PTR_ERR(sai->clk_x8k) != -EPROBE_DEFER)158158+ dev_err(dev, "missing x8k parent clock: %ld\n",159159+ PTR_ERR(sai->clk_x8k));160160+ return PTR_ERR(sai->clk_x8k);161161+ }162162+163163+ sai->clk_x11k = devm_clk_get(dev, "x11k");164164+ if (IS_ERR(sai->clk_x11k)) {165165+ if (PTR_ERR(sai->clk_x11k) != -EPROBE_DEFER)166166+ dev_err(dev, "missing x11k parent clock: %ld\n",167167+ PTR_ERR(sai->clk_x11k));168168+ return PTR_ERR(sai->clk_x11k);169169+ }170170+171171+ return 0;172172+}173173+167174static int stm32_sai_probe(struct platform_device *pdev)168175{169176 struct stm32_sai_data *sai;···198159 sai = devm_kzalloc(&pdev->dev, sizeof(*sai), GFP_KERNEL);199160 if (!sai)200161 return -ENOMEM;162162+163163+ sai->pdev = pdev;201164202165 sai->base = devm_platform_ioremap_resource(pdev, 0);203166 if (IS_ERR(sai->base))···219178 "missing bus clock pclk\n");220179 }221180222222- sai->clk_x8k = devm_clk_get(&pdev->dev, "x8k");223223- if (IS_ERR(sai->clk_x8k))224224- return dev_err_probe(&pdev->dev, PTR_ERR(sai->clk_x8k),225225- "missing x8k parent clock\n");226226-227227- sai->clk_x11k = devm_clk_get(&pdev->dev, "x11k");228228- if (IS_ERR(sai->clk_x11k))229229- return dev_err_probe(&pdev->dev, PTR_ERR(sai->clk_x11k),230230- "missing x11k parent clock\n");181181+ if (sai->conf.get_sai_ck_parent) {182182+ ret = sai->conf.get_sai_ck_parent(sai);183183+ if (ret)184184+ return ret;185185+ }231186232187 /* init irqs */233188 sai->irq = platform_get_irq(pdev, 0);···264227 }265228 clk_disable_unprepare(sai->pclk);266229267267- sai->pdev = pdev;268230 sai->set_sync = &stm32_sai_set_sync;269231 platform_set_drvdata(pdev, sai);270232
+6
sound/soc/stm/stm32_sai.h
···264264 STM_SAI_SYNC_OUT_B,265265};266266267267+struct stm32_sai_data;268268+267269/**268270 * struct stm32_sai_conf - SAI configuration271271+ * @get_sai_ck_parent: get parent clock of SAI kernel clock269272 * @version: SAI version270273 * @fifo_size: SAI fifo size as words number271274 * @has_spdif_pdm: SAI S/PDIF and PDM features support flag275275+ * @no_dma_burst: Support only DMA single transfers if set272276 */273277struct stm32_sai_conf {278278+ int (*get_sai_ck_parent)(struct stm32_sai_data *sai);274279 u32 version;275280 u32 fifo_size;276281 bool has_spdif_pdm;282282+ bool no_dma_burst;277283};278284279285/**
+138-6
sound/soc/stm/stm32_sai_sub.c
···60606161#define SAI_MCLK_NAME_LEN 326262#define SAI_RATE_11K 110256363+#define SAI_MAX_SAMPLE_RATE_8K 1920006464+#define SAI_MAX_SAMPLE_RATE_11K 1764006565+#define SAI_CK_RATE_TOLERANCE 1000 /* ppm */63666467/**6568 * struct stm32_sai_sub_data - private data of SAI sub block (block A or B)···8380 * @dir: SAI block direction (playback or capture). set at init8481 * @master: SAI block mode flag. (true=master, false=slave) set at init8582 * @spdif: SAI S/PDIF iec60958 mode flag. set at init8383+ * @sai_ck_used: flag set while exclusivity on SAI kernel clock is active8684 * @fmt: SAI block format. relevant only for custom protocols. set at init8785 * @sync: SAI block synchronization mode. (none, internal or external)8886 * @synco: SAI block ext sync source (provider setting). (none, sub-block A/B)···9793 * @iec958: iec958 data9894 * @ctrl_lock: control lock9995 * @irq_lock: prevent race condition with IRQ9696+ * @set_sai_ck_rate: set SAI kernel clock rate9797+ * @put_sai_ck_rate: put SAI kernel clock rate10098 */10199struct stm32_sai_sub_data {102100 struct platform_device *pdev;···118112 int dir;119113 bool master;120114 bool spdif;115115+ bool sai_ck_used;121116 int fmt;122117 int sync;123118 int synco;···132125 struct snd_aes_iec958 iec958;133126 struct mutex ctrl_lock; /* protect resources accessed by controls */134127 spinlock_t irq_lock; /* used to prevent race condition with IRQ */128128+ int (*set_sai_ck_rate)(struct stm32_sai_sub_data *sai, unsigned int rate);129129+ void (*put_sai_ck_rate)(struct stm32_sai_sub_data *sai);135130};136131137132enum stm32_sai_fifo_th {···360351 return ret;361352}362353363363-static int stm32_sai_set_parent_clock(struct stm32_sai_sub_data *sai,364364- unsigned int rate)354354+static bool stm32_sai_rate_accurate(unsigned int max_rate, unsigned int rate)355355+{356356+ u64 delta, dividend;357357+ int ratio;358358+359359+ ratio = DIV_ROUND_CLOSEST(max_rate, rate);360360+ if (!ratio)361361+ return false;362362+363363+ dividend = mul_u32_u32(1000000, abs(max_rate - (ratio * rate)));364364+ delta = div_u64(dividend, max_rate);365365+366366+ if (delta <= SAI_CK_RATE_TOLERANCE)367367+ return true;368368+369369+ return false;370370+}371371+372372+static int stm32_sai_set_parent_clk(struct stm32_sai_sub_data *sai,373373+ unsigned int rate)365374{366375 struct platform_device *pdev = sai->pdev;367376 struct clk *parent_clk = sai->pdata->clk_x8k;···395368 "Active stream rates conflict\n" : "\n");396369397370 return ret;371371+}372372+373373+static void stm32_sai_put_parent_rate(struct stm32_sai_sub_data *sai)374374+{375375+ if (sai->sai_ck_used) {376376+ sai->sai_ck_used = false;377377+ clk_rate_exclusive_put(sai->sai_ck);378378+ }379379+}380380+381381+static int stm32_sai_set_parent_rate(struct stm32_sai_sub_data *sai,382382+ unsigned int rate)383383+{384384+ struct platform_device *pdev = sai->pdev;385385+ unsigned int sai_ck_rate, sai_ck_max_rate, sai_curr_rate, sai_new_rate;386386+ int div, ret;387387+388388+ /*389389+ * Set maximum expected kernel clock frequency390390+ * - mclk on or spdif:391391+ * f_sai_ck = MCKDIV * mclk-fs * fs392392+ * Here typical 256 ratio is assumed for mclk-fs393393+ * - mclk off:394394+ * f_sai_ck = MCKDIV * FRL * fs395395+ * Where FRL=[8..256], MCKDIV=[1..n] (n depends on SAI version)396396+ * Set constraint MCKDIV * FRL <= 256, to ensure MCKDIV is in available range397397+ * f_sai_ck = sai_ck_max_rate * pow_of_two(FRL) / 256398398+ */399399+ if (!(rate % SAI_RATE_11K))400400+ sai_ck_max_rate = SAI_MAX_SAMPLE_RATE_11K * 256;401401+ else402402+ sai_ck_max_rate = SAI_MAX_SAMPLE_RATE_8K * 256;403403+404404+ if (!sai->sai_mclk && !STM_SAI_PROTOCOL_IS_SPDIF(sai))405405+ sai_ck_max_rate /= DIV_ROUND_CLOSEST(256, roundup_pow_of_two(sai->fs_length));406406+407407+ /*408408+ * Request exclusivity, as the clock is shared by SAI sub-blocks and by409409+ * some SAI instances. This allows to ensure that the rate cannot be410410+ * changed while one or more SAIs are using the clock.411411+ */412412+ clk_rate_exclusive_get(sai->sai_ck);413413+ sai->sai_ck_used = true;414414+415415+ /*416416+ * Check current kernel clock rate. If it gives the expected accuracy417417+ * return immediately.418418+ */419419+ sai_curr_rate = clk_get_rate(sai->sai_ck);420420+ if (stm32_sai_rate_accurate(sai_ck_max_rate, sai_curr_rate))421421+ return 0;422422+423423+ /*424424+ * Otherwise try to set the maximum rate and check the new actual rate.425425+ * If the new rate does not give the expected accuracy, try to set426426+ * lower rates for the kernel clock.427427+ */428428+ sai_ck_rate = sai_ck_max_rate;429429+ div = 1;430430+ do {431431+ /* Check new rate accuracy. Return if ok */432432+ sai_new_rate = clk_round_rate(sai->sai_ck, sai_ck_rate);433433+ if (stm32_sai_rate_accurate(sai_ck_rate, sai_new_rate)) {434434+ ret = clk_set_rate(sai->sai_ck, sai_ck_rate);435435+ if (ret) {436436+ dev_err(&pdev->dev, "Error %d setting sai_ck rate. %s",437437+ ret, ret == -EBUSY ?438438+ "Active stream rates may be in conflict\n" : "\n");439439+ goto err;440440+ }441441+442442+ return 0;443443+ }444444+445445+ /* Try a lower frequency */446446+ div++;447447+ sai_ck_rate = sai_ck_max_rate / div;448448+ } while (sai_ck_rate > rate);449449+450450+ /* No accurate rate found */451451+ dev_err(&pdev->dev, "Failed to find an accurate rate");452452+453453+err:454454+ stm32_sai_put_parent_rate(sai);455455+456456+ return -EINVAL;398457}399458400459static long stm32_sai_mclk_round_rate(struct clk_hw *hw, unsigned long rate,···678565 clk_rate_exclusive_put(sai->sai_mclk);679566 sai->mclk_rate = 0;680567 }568568+569569+ if (sai->put_sai_ck_rate)570570+ sai->put_sai_ck_rate(sai);571571+681572 return 0;682573 }683574684684- /* If master clock is used, set parent clock now */685685- ret = stm32_sai_set_parent_clock(sai, freq);575575+ /* If master clock is used, configure SAI kernel clock now */576576+ ret = sai->set_sai_ck_rate(sai, freq);686577 if (ret)687578 return ret;688579···1110993 int ret;11119941112995 if (!sai->sai_mclk) {11131113- ret = stm32_sai_set_parent_clock(sai, rate);996996+ ret = sai->set_sai_ck_rate(sai, rate);1114997 if (ret)1115998 return ret;1116999 }···1271115412721155 clk_disable_unprepare(sai->sai_ck);1273115611571157+ /*11581158+ * Release kernel clock if following conditions are fulfilled11591159+ * - Master clock is not used. Kernel clock won't be released trough sysclk11601160+ * - Put handler is defined. Involve that clock is managed exclusively11611161+ */11621162+ if (!sai->sai_mclk && sai->put_sai_ck_rate)11631163+ sai->put_sai_ck_rate(sai);11641164+12741165 spin_lock_irqsave(&sai->irq_lock, flags);12751166 sai->substream = NULL;12761167 spin_unlock_irqrestore(&sai->irq_lock, flags);···13131188 * constraints).13141189 */13151190 sai->dma_params.maxburst = 4;13161316- if (sai->pdata->conf.fifo_size < 8)11911191+ if (sai->pdata->conf.fifo_size < 8 || sai->pdata->conf.no_dma_burst)13171192 sai->dma_params.maxburst = 1;13181193 /* Buswidth will be set by framework at runtime */13191194 sai->dma_params.addr_width = DMA_SLAVE_BUSWIDTH_UNDEFINED;···16491524 if (!sai->pdata) {16501525 dev_err(&pdev->dev, "Parent device data not available\n");16511526 return -EINVAL;15271527+ }15281528+15291529+ if (sai->pdata->conf.get_sai_ck_parent) {15301530+ sai->set_sai_ck_rate = stm32_sai_set_parent_clk;15311531+ } else {15321532+ sai->set_sai_ck_rate = stm32_sai_set_parent_rate;15331533+ sai->put_sai_ck_rate = stm32_sai_put_parent_rate;16521534 }1653153516541536 ret = stm32_sai_sub_parse_of(pdev, sai);