diff options
-rw-r--r-- | sound/soc/fsl/fsl_ssi.c | 185 |
1 files changed, 103 insertions, 82 deletions
diff --git a/sound/soc/fsl/fsl_ssi.c b/sound/soc/fsl/fsl_ssi.c index 65d9da190d0c..418c64629ee7 100644 --- a/sound/soc/fsl/fsl_ssi.c +++ b/sound/soc/fsl/fsl_ssi.c | |||
@@ -992,6 +992,100 @@ static void make_lowercase(char *s) | |||
992 | } | 992 | } |
993 | } | 993 | } |
994 | 994 | ||
995 | static int fsl_ssi_imx_probe(struct platform_device *pdev, | ||
996 | struct fsl_ssi_private *ssi_private) | ||
997 | { | ||
998 | struct device_node *np = pdev->dev.of_node; | ||
999 | u32 dma_events[2], dmas[4]; | ||
1000 | bool shared; | ||
1001 | int ret; | ||
1002 | |||
1003 | ssi_private->clk = devm_clk_get(&pdev->dev, NULL); | ||
1004 | if (IS_ERR(ssi_private->clk)) { | ||
1005 | ret = PTR_ERR(ssi_private->clk); | ||
1006 | dev_err(&pdev->dev, "could not get clock: %d\n", ret); | ||
1007 | return ret; | ||
1008 | } | ||
1009 | |||
1010 | ret = clk_prepare_enable(ssi_private->clk); | ||
1011 | if (ret) { | ||
1012 | dev_err(&pdev->dev, "clk_prepare_enable failed: %d\n", ret); | ||
1013 | return ret; | ||
1014 | } | ||
1015 | |||
1016 | /* For those SLAVE implementations, we ingore non-baudclk cases | ||
1017 | * and, instead, abandon MASTER mode that needs baud clock. | ||
1018 | */ | ||
1019 | ssi_private->baudclk = devm_clk_get(&pdev->dev, "baud"); | ||
1020 | if (IS_ERR(ssi_private->baudclk)) | ||
1021 | dev_dbg(&pdev->dev, "could not get baud clock: %ld\n", | ||
1022 | PTR_ERR(ssi_private->baudclk)); | ||
1023 | else | ||
1024 | clk_prepare_enable(ssi_private->baudclk); | ||
1025 | |||
1026 | /* | ||
1027 | * We have burstsize be "fifo_depth - 2" to match the SSI | ||
1028 | * watermark setting in fsl_ssi_startup(). | ||
1029 | */ | ||
1030 | ssi_private->dma_params_tx.maxburst = ssi_private->fifo_depth - 2; | ||
1031 | ssi_private->dma_params_rx.maxburst = ssi_private->fifo_depth - 2; | ||
1032 | ssi_private->dma_params_tx.addr = ssi_private->ssi_phys + | ||
1033 | offsetof(struct ccsr_ssi, stx0); | ||
1034 | ssi_private->dma_params_rx.addr = ssi_private->ssi_phys + | ||
1035 | offsetof(struct ccsr_ssi, srx0); | ||
1036 | ssi_private->dma_params_tx.filter_data = &ssi_private->filter_data_tx; | ||
1037 | ssi_private->dma_params_rx.filter_data = &ssi_private->filter_data_rx; | ||
1038 | |||
1039 | if (!of_property_read_bool(pdev->dev.of_node, "dmas") && | ||
1040 | ssi_private->use_dma) { | ||
1041 | /* | ||
1042 | * FIXME: This is a temporary solution until all | ||
1043 | * necessary dma drivers support the generic dma | ||
1044 | * bindings. | ||
1045 | */ | ||
1046 | ret = of_property_read_u32_array(pdev->dev.of_node, | ||
1047 | "fsl,ssi-dma-events", dma_events, 2); | ||
1048 | if (ret && ssi_private->use_dma) { | ||
1049 | dev_err(&pdev->dev, "could not get dma events but fsl-ssi is configured to use DMA\n"); | ||
1050 | goto error_dma_events; | ||
1051 | } | ||
1052 | } | ||
1053 | /* Should this be merge with the above? */ | ||
1054 | if (!of_property_read_u32_array(pdev->dev.of_node, "dmas", dmas, 4) | ||
1055 | && dmas[2] == IMX_DMATYPE_SSI_DUAL) { | ||
1056 | ssi_private->use_dual_fifo = true; | ||
1057 | /* When using dual fifo mode, we need to keep watermark | ||
1058 | * as even numbers due to dma script limitation. | ||
1059 | */ | ||
1060 | ssi_private->dma_params_tx.maxburst &= ~0x1; | ||
1061 | ssi_private->dma_params_rx.maxburst &= ~0x1; | ||
1062 | } | ||
1063 | |||
1064 | shared = of_device_is_compatible(of_get_parent(np), "fsl,spba-bus"); | ||
1065 | |||
1066 | imx_pcm_dma_params_init_data(&ssi_private->filter_data_tx, | ||
1067 | dma_events[0], shared ? IMX_DMATYPE_SSI_SP : IMX_DMATYPE_SSI); | ||
1068 | imx_pcm_dma_params_init_data(&ssi_private->filter_data_rx, | ||
1069 | dma_events[1], shared ? IMX_DMATYPE_SSI_SP : IMX_DMATYPE_SSI); | ||
1070 | |||
1071 | return 0; | ||
1072 | |||
1073 | error_dma_events: | ||
1074 | if (!IS_ERR(ssi_private->baudclk)) | ||
1075 | clk_disable_unprepare(ssi_private->baudclk); | ||
1076 | clk_disable_unprepare(ssi_private->clk); | ||
1077 | |||
1078 | return ret; | ||
1079 | } | ||
1080 | |||
1081 | static void fsl_ssi_imx_clean(struct platform_device *pdev, | ||
1082 | struct fsl_ssi_private *ssi_private) | ||
1083 | { | ||
1084 | if (!IS_ERR(ssi_private->baudclk)) | ||
1085 | clk_disable_unprepare(ssi_private->baudclk); | ||
1086 | clk_disable_unprepare(ssi_private->clk); | ||
1087 | } | ||
1088 | |||
995 | static int fsl_ssi_probe(struct platform_device *pdev) | 1089 | static int fsl_ssi_probe(struct platform_device *pdev) |
996 | { | 1090 | { |
997 | struct fsl_ssi_private *ssi_private; | 1091 | struct fsl_ssi_private *ssi_private; |
@@ -1004,7 +1098,6 @@ static int fsl_ssi_probe(struct platform_device *pdev) | |||
1004 | const uint32_t *iprop; | 1098 | const uint32_t *iprop; |
1005 | struct resource res; | 1099 | struct resource res; |
1006 | char name[64]; | 1100 | char name[64]; |
1007 | bool shared; | ||
1008 | bool ac97 = false; | 1101 | bool ac97 = false; |
1009 | 1102 | ||
1010 | /* SSIs that are not connected on the board should have a | 1103 | /* SSIs that are not connected on the board should have a |
@@ -1118,80 +1211,11 @@ static int fsl_ssi_probe(struct platform_device *pdev) | |||
1118 | 1211 | ||
1119 | if (hw_type == FSL_SSI_MX21 || hw_type == FSL_SSI_MX51 || | 1212 | if (hw_type == FSL_SSI_MX21 || hw_type == FSL_SSI_MX51 || |
1120 | hw_type == FSL_SSI_MX35) { | 1213 | hw_type == FSL_SSI_MX35) { |
1121 | u32 dma_events[2], dmas[4]; | ||
1122 | ssi_private->ssi_on_imx = true; | 1214 | ssi_private->ssi_on_imx = true; |
1123 | 1215 | ||
1124 | ssi_private->clk = devm_clk_get(&pdev->dev, NULL); | 1216 | ret = fsl_ssi_imx_probe(pdev, ssi_private); |
1125 | if (IS_ERR(ssi_private->clk)) { | 1217 | if (ret) |
1126 | ret = PTR_ERR(ssi_private->clk); | ||
1127 | dev_err(&pdev->dev, "could not get clock: %d\n", ret); | ||
1128 | goto error_irqmap; | 1218 | goto error_irqmap; |
1129 | } | ||
1130 | ret = clk_prepare_enable(ssi_private->clk); | ||
1131 | if (ret) { | ||
1132 | dev_err(&pdev->dev, "clk_prepare_enable failed: %d\n", | ||
1133 | ret); | ||
1134 | goto error_irqmap; | ||
1135 | } | ||
1136 | |||
1137 | /* For those SLAVE implementations, we ingore non-baudclk cases | ||
1138 | * and, instead, abandon MASTER mode that needs baud clock. | ||
1139 | */ | ||
1140 | ssi_private->baudclk = devm_clk_get(&pdev->dev, "baud"); | ||
1141 | if (IS_ERR(ssi_private->baudclk)) | ||
1142 | dev_dbg(&pdev->dev, "could not get baud clock: %ld\n", | ||
1143 | PTR_ERR(ssi_private->baudclk)); | ||
1144 | else | ||
1145 | clk_prepare_enable(ssi_private->baudclk); | ||
1146 | |||
1147 | /* | ||
1148 | * We have burstsize be "fifo_depth - 2" to match the SSI | ||
1149 | * watermark setting in fsl_ssi_startup(). | ||
1150 | */ | ||
1151 | ssi_private->dma_params_tx.maxburst = | ||
1152 | ssi_private->fifo_depth - 2; | ||
1153 | ssi_private->dma_params_rx.maxburst = | ||
1154 | ssi_private->fifo_depth - 2; | ||
1155 | ssi_private->dma_params_tx.addr = | ||
1156 | ssi_private->ssi_phys + offsetof(struct ccsr_ssi, stx0); | ||
1157 | ssi_private->dma_params_rx.addr = | ||
1158 | ssi_private->ssi_phys + offsetof(struct ccsr_ssi, srx0); | ||
1159 | ssi_private->dma_params_tx.filter_data = | ||
1160 | &ssi_private->filter_data_tx; | ||
1161 | ssi_private->dma_params_rx.filter_data = | ||
1162 | &ssi_private->filter_data_rx; | ||
1163 | if (!of_property_read_bool(pdev->dev.of_node, "dmas") && | ||
1164 | ssi_private->use_dma) { | ||
1165 | /* | ||
1166 | * FIXME: This is a temporary solution until all | ||
1167 | * necessary dma drivers support the generic dma | ||
1168 | * bindings. | ||
1169 | */ | ||
1170 | ret = of_property_read_u32_array(pdev->dev.of_node, | ||
1171 | "fsl,ssi-dma-events", dma_events, 2); | ||
1172 | if (ret && ssi_private->use_dma) { | ||
1173 | dev_err(&pdev->dev, "could not get dma events but fsl-ssi is configured to use DMA\n"); | ||
1174 | goto error_clk; | ||
1175 | } | ||
1176 | } | ||
1177 | /* Should this be merge with the above? */ | ||
1178 | if (!of_property_read_u32_array(pdev->dev.of_node, "dmas", dmas, 4) | ||
1179 | && dmas[2] == IMX_DMATYPE_SSI_DUAL) { | ||
1180 | ssi_private->use_dual_fifo = true; | ||
1181 | /* When using dual fifo mode, we need to keep watermark | ||
1182 | * as even numbers due to dma script limitation. | ||
1183 | */ | ||
1184 | ssi_private->dma_params_tx.maxburst &= ~0x1; | ||
1185 | ssi_private->dma_params_rx.maxburst &= ~0x1; | ||
1186 | } | ||
1187 | |||
1188 | shared = of_device_is_compatible(of_get_parent(np), | ||
1189 | "fsl,spba-bus"); | ||
1190 | |||
1191 | imx_pcm_dma_params_init_data(&ssi_private->filter_data_tx, | ||
1192 | dma_events[0], shared ? IMX_DMATYPE_SSI_SP : IMX_DMATYPE_SSI); | ||
1193 | imx_pcm_dma_params_init_data(&ssi_private->filter_data_rx, | ||
1194 | dma_events[1], shared ? IMX_DMATYPE_SSI_SP : IMX_DMATYPE_SSI); | ||
1195 | } | 1219 | } |
1196 | 1220 | ||
1197 | /* | 1221 | /* |
@@ -1206,7 +1230,7 @@ static int fsl_ssi_probe(struct platform_device *pdev) | |||
1206 | if (ret < 0) { | 1230 | if (ret < 0) { |
1207 | dev_err(&pdev->dev, "could not claim irq %u\n", | 1231 | dev_err(&pdev->dev, "could not claim irq %u\n", |
1208 | ssi_private->irq); | 1232 | ssi_private->irq); |
1209 | goto error_clk; | 1233 | goto error_irq; |
1210 | } | 1234 | } |
1211 | } | 1235 | } |
1212 | 1236 | ||
@@ -1298,11 +1322,9 @@ error_dbgfs: | |||
1298 | error_dev: | 1322 | error_dev: |
1299 | device_remove_file(&pdev->dev, dev_attr); | 1323 | device_remove_file(&pdev->dev, dev_attr); |
1300 | 1324 | ||
1301 | error_clk: | 1325 | error_irq: |
1302 | if (ssi_private->ssi_on_imx) { | 1326 | if (ssi_private->ssi_on_imx) { |
1303 | if (!IS_ERR(ssi_private->baudclk)) | 1327 | fsl_ssi_imx_clean(pdev, ssi_private); |
1304 | clk_disable_unprepare(ssi_private->baudclk); | ||
1305 | clk_disable_unprepare(ssi_private->clk); | ||
1306 | } | 1328 | } |
1307 | 1329 | ||
1308 | error_irqmap: | 1330 | error_irqmap: |
@@ -1321,11 +1343,10 @@ static int fsl_ssi_remove(struct platform_device *pdev) | |||
1321 | if (!ssi_private->new_binding) | 1343 | if (!ssi_private->new_binding) |
1322 | platform_device_unregister(ssi_private->pdev); | 1344 | platform_device_unregister(ssi_private->pdev); |
1323 | snd_soc_unregister_component(&pdev->dev); | 1345 | snd_soc_unregister_component(&pdev->dev); |
1324 | if (ssi_private->ssi_on_imx) { | 1346 | |
1325 | if (!IS_ERR(ssi_private->baudclk)) | 1347 | if (ssi_private->ssi_on_imx) |
1326 | clk_disable_unprepare(ssi_private->baudclk); | 1348 | fsl_ssi_imx_clean(pdev, ssi_private); |
1327 | clk_disable_unprepare(ssi_private->clk); | 1349 | |
1328 | } | ||
1329 | if (ssi_private->irq_stats) | 1350 | if (ssi_private->irq_stats) |
1330 | irq_dispose_mapping(ssi_private->irq); | 1351 | irq_dispose_mapping(ssi_private->irq); |
1331 | 1352 | ||