diff options
Diffstat (limited to 'sound/soc/samsung/i2s.c')
-rw-r--r-- | sound/soc/samsung/i2s.c | 267 |
1 files changed, 220 insertions, 47 deletions
diff --git a/sound/soc/samsung/i2s.c b/sound/soc/samsung/i2s.c index d2d124f1dd1b..d7231e336a7c 100644 --- a/sound/soc/samsung/i2s.c +++ b/sound/soc/samsung/i2s.c | |||
@@ -15,11 +15,15 @@ | |||
15 | #include <linux/clk.h> | 15 | #include <linux/clk.h> |
16 | #include <linux/io.h> | 16 | #include <linux/io.h> |
17 | #include <linux/module.h> | 17 | #include <linux/module.h> |
18 | #include <linux/of.h> | ||
19 | #include <linux/of_gpio.h> | ||
18 | #include <linux/pm_runtime.h> | 20 | #include <linux/pm_runtime.h> |
19 | 21 | ||
20 | #include <sound/soc.h> | 22 | #include <sound/soc.h> |
21 | #include <sound/pcm_params.h> | 23 | #include <sound/pcm_params.h> |
22 | 24 | ||
25 | #include <mach/dma.h> | ||
26 | |||
23 | #include <linux/platform_data/asoc-s3c.h> | 27 | #include <linux/platform_data/asoc-s3c.h> |
24 | 28 | ||
25 | #include "dma.h" | 29 | #include "dma.h" |
@@ -29,6 +33,15 @@ | |||
29 | 33 | ||
30 | #define msecs_to_loops(t) (loops_per_jiffy / 1000 * HZ * t) | 34 | #define msecs_to_loops(t) (loops_per_jiffy / 1000 * HZ * t) |
31 | 35 | ||
36 | enum samsung_dai_type { | ||
37 | TYPE_PRI, | ||
38 | TYPE_SEC, | ||
39 | }; | ||
40 | |||
41 | struct samsung_i2s_dai_data { | ||
42 | int dai_type; | ||
43 | }; | ||
44 | |||
32 | struct i2s_dai { | 45 | struct i2s_dai { |
33 | /* Platform device for this DAI */ | 46 | /* Platform device for this DAI */ |
34 | struct platform_device *pdev; | 47 | struct platform_device *pdev; |
@@ -66,6 +79,7 @@ struct i2s_dai { | |||
66 | u32 suspend_i2smod; | 79 | u32 suspend_i2smod; |
67 | u32 suspend_i2scon; | 80 | u32 suspend_i2scon; |
68 | u32 suspend_i2spsr; | 81 | u32 suspend_i2spsr; |
82 | unsigned long gpios[7]; /* i2s gpio line numbers */ | ||
69 | }; | 83 | }; |
70 | 84 | ||
71 | /* Lock for cross i/f checks */ | 85 | /* Lock for cross i/f checks */ |
@@ -651,6 +665,9 @@ static int i2s_startup(struct snd_pcm_substream *substream, | |||
651 | /* Enforce set_sysclk in Master mode */ | 665 | /* Enforce set_sysclk in Master mode */ |
652 | i2s->rclk_srcrate = 0; | 666 | i2s->rclk_srcrate = 0; |
653 | 667 | ||
668 | if (!any_active(i2s) && (i2s->quirks & QUIRK_NEED_RSTCLR)) | ||
669 | writel(CON_RSTCLR, i2s->addr + I2SCON); | ||
670 | |||
654 | spin_unlock_irqrestore(&lock, flags); | 671 | spin_unlock_irqrestore(&lock, flags); |
655 | 672 | ||
656 | return 0; | 673 | return 0; |
@@ -981,8 +998,7 @@ static struct i2s_dai *i2s_alloc_dai(struct platform_device *pdev, bool sec) | |||
981 | i2s->i2s_dai_drv.capture.formats = SAMSUNG_I2S_FMTS; | 998 | i2s->i2s_dai_drv.capture.formats = SAMSUNG_I2S_FMTS; |
982 | } else { /* Create a new platform_device for Secondary */ | 999 | } else { /* Create a new platform_device for Secondary */ |
983 | i2s->pdev = platform_device_register_resndata(NULL, | 1000 | i2s->pdev = platform_device_register_resndata(NULL, |
984 | pdev->name, pdev->id + SAMSUNG_I2S_SECOFF, | 1001 | "samsung-i2s-sec", -1, NULL, 0, NULL, 0); |
985 | NULL, 0, NULL, 0); | ||
986 | if (IS_ERR(i2s->pdev)) | 1002 | if (IS_ERR(i2s->pdev)) |
987 | return NULL; | 1003 | return NULL; |
988 | } | 1004 | } |
@@ -993,18 +1009,103 @@ static struct i2s_dai *i2s_alloc_dai(struct platform_device *pdev, bool sec) | |||
993 | return i2s; | 1009 | return i2s; |
994 | } | 1010 | } |
995 | 1011 | ||
1012 | #ifdef CONFIG_OF | ||
1013 | static int samsung_i2s_parse_dt_gpio(struct i2s_dai *i2s) | ||
1014 | { | ||
1015 | struct device *dev = &i2s->pdev->dev; | ||
1016 | int index, gpio, ret; | ||
1017 | |||
1018 | for (index = 0; index < 7; index++) { | ||
1019 | gpio = of_get_gpio(dev->of_node, index); | ||
1020 | if (!gpio_is_valid(gpio)) { | ||
1021 | dev_err(dev, "invalid gpio[%d]: %d\n", index, gpio); | ||
1022 | goto free_gpio; | ||
1023 | } | ||
1024 | |||
1025 | ret = gpio_request(gpio, dev_name(dev)); | ||
1026 | if (ret) { | ||
1027 | dev_err(dev, "gpio [%d] request failed\n", gpio); | ||
1028 | goto free_gpio; | ||
1029 | } | ||
1030 | i2s->gpios[index] = gpio; | ||
1031 | } | ||
1032 | return 0; | ||
1033 | |||
1034 | free_gpio: | ||
1035 | while (--index >= 0) | ||
1036 | gpio_free(i2s->gpios[index]); | ||
1037 | return -EINVAL; | ||
1038 | } | ||
1039 | |||
1040 | static void samsung_i2s_dt_gpio_free(struct i2s_dai *i2s) | ||
1041 | { | ||
1042 | unsigned int index; | ||
1043 | for (index = 0; index < 7; index++) | ||
1044 | gpio_free(i2s->gpios[index]); | ||
1045 | } | ||
1046 | #else | ||
1047 | static int samsung_i2s_parse_dt_gpio(struct i2s_dai *dai) | ||
1048 | { | ||
1049 | return -EINVAL; | ||
1050 | } | ||
1051 | |||
1052 | static void samsung_i2s_dt_gpio_free(struct i2s_dai *dai) | ||
1053 | { | ||
1054 | } | ||
1055 | |||
1056 | #endif | ||
1057 | |||
1058 | static const struct of_device_id exynos_i2s_match[]; | ||
1059 | |||
1060 | static inline int samsung_i2s_get_driver_data(struct platform_device *pdev) | ||
1061 | { | ||
1062 | #ifdef CONFIG_OF | ||
1063 | struct samsung_i2s_dai_data *data; | ||
1064 | if (pdev->dev.of_node) { | ||
1065 | const struct of_device_id *match; | ||
1066 | match = of_match_node(exynos_i2s_match, pdev->dev.of_node); | ||
1067 | data = (struct samsung_i2s_dai_data *) match->data; | ||
1068 | return data->dai_type; | ||
1069 | } else | ||
1070 | #endif | ||
1071 | return platform_get_device_id(pdev)->driver_data; | ||
1072 | } | ||
1073 | |||
1074 | #ifdef CONFIG_PM_RUNTIME | ||
1075 | static int i2s_runtime_suspend(struct device *dev) | ||
1076 | { | ||
1077 | struct i2s_dai *i2s = dev_get_drvdata(dev); | ||
1078 | |||
1079 | clk_disable_unprepare(i2s->clk); | ||
1080 | |||
1081 | return 0; | ||
1082 | } | ||
1083 | |||
1084 | static int i2s_runtime_resume(struct device *dev) | ||
1085 | { | ||
1086 | struct i2s_dai *i2s = dev_get_drvdata(dev); | ||
1087 | |||
1088 | clk_prepare_enable(i2s->clk); | ||
1089 | |||
1090 | return 0; | ||
1091 | } | ||
1092 | #endif /* CONFIG_PM_RUNTIME */ | ||
1093 | |||
996 | static int samsung_i2s_probe(struct platform_device *pdev) | 1094 | static int samsung_i2s_probe(struct platform_device *pdev) |
997 | { | 1095 | { |
998 | u32 dma_pl_chan, dma_cp_chan, dma_pl_sec_chan; | ||
999 | struct i2s_dai *pri_dai, *sec_dai = NULL; | 1096 | struct i2s_dai *pri_dai, *sec_dai = NULL; |
1000 | struct s3c_audio_pdata *i2s_pdata; | 1097 | struct s3c_audio_pdata *i2s_pdata = pdev->dev.platform_data; |
1001 | struct samsung_i2s *i2s_cfg; | 1098 | struct samsung_i2s *i2s_cfg = NULL; |
1002 | struct resource *res; | 1099 | struct resource *res; |
1003 | u32 regs_base, quirks; | 1100 | u32 regs_base, quirks = 0, idma_addr = 0; |
1101 | struct device_node *np = pdev->dev.of_node; | ||
1102 | enum samsung_dai_type samsung_dai_type; | ||
1004 | int ret = 0; | 1103 | int ret = 0; |
1005 | 1104 | ||
1006 | /* Call during Seconday interface registration */ | 1105 | /* Call during Seconday interface registration */ |
1007 | if (pdev->id >= SAMSUNG_I2S_SECOFF) { | 1106 | samsung_dai_type = samsung_i2s_get_driver_data(pdev); |
1107 | |||
1108 | if (samsung_dai_type == TYPE_SEC) { | ||
1008 | sec_dai = dev_get_drvdata(&pdev->dev); | 1109 | sec_dai = dev_get_drvdata(&pdev->dev); |
1009 | snd_soc_register_dai(&sec_dai->pdev->dev, | 1110 | snd_soc_register_dai(&sec_dai->pdev->dev, |
1010 | &sec_dai->i2s_dai_drv); | 1111 | &sec_dai->i2s_dai_drv); |
@@ -1012,31 +1113,60 @@ static int samsung_i2s_probe(struct platform_device *pdev) | |||
1012 | return 0; | 1113 | return 0; |
1013 | } | 1114 | } |
1014 | 1115 | ||
1015 | i2s_pdata = pdev->dev.platform_data; | 1116 | pri_dai = i2s_alloc_dai(pdev, false); |
1016 | if (i2s_pdata == NULL) { | 1117 | if (!pri_dai) { |
1017 | dev_err(&pdev->dev, "Can't work without s3c_audio_pdata\n"); | 1118 | dev_err(&pdev->dev, "Unable to alloc I2S_pri\n"); |
1018 | return -EINVAL; | 1119 | return -ENOMEM; |
1019 | } | 1120 | } |
1020 | 1121 | ||
1021 | res = platform_get_resource(pdev, IORESOURCE_DMA, 0); | 1122 | if (!np) { |
1022 | if (!res) { | 1123 | res = platform_get_resource(pdev, IORESOURCE_DMA, 0); |
1023 | dev_err(&pdev->dev, "Unable to get I2S-TX dma resource\n"); | 1124 | if (!res) { |
1024 | return -ENXIO; | 1125 | dev_err(&pdev->dev, |
1025 | } | 1126 | "Unable to get I2S-TX dma resource\n"); |
1026 | dma_pl_chan = res->start; | 1127 | return -ENXIO; |
1128 | } | ||
1129 | pri_dai->dma_playback.channel = res->start; | ||
1027 | 1130 | ||
1028 | res = platform_get_resource(pdev, IORESOURCE_DMA, 1); | 1131 | res = platform_get_resource(pdev, IORESOURCE_DMA, 1); |
1029 | if (!res) { | 1132 | if (!res) { |
1030 | dev_err(&pdev->dev, "Unable to get I2S-RX dma resource\n"); | 1133 | dev_err(&pdev->dev, |
1031 | return -ENXIO; | 1134 | "Unable to get I2S-RX dma resource\n"); |
1032 | } | 1135 | return -ENXIO; |
1033 | dma_cp_chan = res->start; | 1136 | } |
1137 | pri_dai->dma_capture.channel = res->start; | ||
1034 | 1138 | ||
1035 | res = platform_get_resource(pdev, IORESOURCE_DMA, 2); | 1139 | if (i2s_pdata == NULL) { |
1036 | if (res) | 1140 | dev_err(&pdev->dev, "Can't work without s3c_audio_pdata\n"); |
1037 | dma_pl_sec_chan = res->start; | 1141 | return -EINVAL; |
1038 | else | 1142 | } |
1039 | dma_pl_sec_chan = 0; | 1143 | |
1144 | if (&i2s_pdata->type) | ||
1145 | i2s_cfg = &i2s_pdata->type.i2s; | ||
1146 | |||
1147 | if (i2s_cfg) { | ||
1148 | quirks = i2s_cfg->quirks; | ||
1149 | idma_addr = i2s_cfg->idma_addr; | ||
1150 | } | ||
1151 | } else { | ||
1152 | if (of_find_property(np, "samsung,supports-6ch", NULL)) | ||
1153 | quirks |= QUIRK_PRI_6CHAN; | ||
1154 | |||
1155 | if (of_find_property(np, "samsung,supports-secdai", NULL)) | ||
1156 | quirks |= QUIRK_SEC_DAI; | ||
1157 | |||
1158 | if (of_find_property(np, "samsung,supports-rstclr", NULL)) | ||
1159 | quirks |= QUIRK_NEED_RSTCLR; | ||
1160 | |||
1161 | if (of_property_read_u32(np, "samsung,idma-addr", | ||
1162 | &idma_addr)) { | ||
1163 | if (quirks & QUIRK_SEC_DAI) { | ||
1164 | dev_err(&pdev->dev, "idma address is not"\ | ||
1165 | "specified"); | ||
1166 | return -EINVAL; | ||
1167 | } | ||
1168 | } | ||
1169 | } | ||
1040 | 1170 | ||
1041 | res = platform_get_resource(pdev, IORESOURCE_MEM, 0); | 1171 | res = platform_get_resource(pdev, IORESOURCE_MEM, 0); |
1042 | if (!res) { | 1172 | if (!res) { |
@@ -1051,24 +1181,14 @@ static int samsung_i2s_probe(struct platform_device *pdev) | |||
1051 | } | 1181 | } |
1052 | regs_base = res->start; | 1182 | regs_base = res->start; |
1053 | 1183 | ||
1054 | i2s_cfg = &i2s_pdata->type.i2s; | ||
1055 | quirks = i2s_cfg->quirks; | ||
1056 | |||
1057 | pri_dai = i2s_alloc_dai(pdev, false); | ||
1058 | if (!pri_dai) { | ||
1059 | dev_err(&pdev->dev, "Unable to alloc I2S_pri\n"); | ||
1060 | ret = -ENOMEM; | ||
1061 | goto err; | ||
1062 | } | ||
1063 | |||
1064 | pri_dai->dma_playback.dma_addr = regs_base + I2STXD; | 1184 | pri_dai->dma_playback.dma_addr = regs_base + I2STXD; |
1065 | pri_dai->dma_capture.dma_addr = regs_base + I2SRXD; | 1185 | pri_dai->dma_capture.dma_addr = regs_base + I2SRXD; |
1066 | pri_dai->dma_playback.client = | 1186 | pri_dai->dma_playback.client = |
1067 | (struct s3c2410_dma_client *)&pri_dai->dma_playback; | 1187 | (struct s3c2410_dma_client *)&pri_dai->dma_playback; |
1188 | pri_dai->dma_playback.ch_name = "tx"; | ||
1068 | pri_dai->dma_capture.client = | 1189 | pri_dai->dma_capture.client = |
1069 | (struct s3c2410_dma_client *)&pri_dai->dma_capture; | 1190 | (struct s3c2410_dma_client *)&pri_dai->dma_capture; |
1070 | pri_dai->dma_playback.channel = dma_pl_chan; | 1191 | pri_dai->dma_capture.ch_name = "rx"; |
1071 | pri_dai->dma_capture.channel = dma_cp_chan; | ||
1072 | pri_dai->dma_playback.dma_size = 4; | 1192 | pri_dai->dma_playback.dma_size = 4; |
1073 | pri_dai->dma_capture.dma_size = 4; | 1193 | pri_dai->dma_capture.dma_size = 4; |
1074 | pri_dai->base = regs_base; | 1194 | pri_dai->base = regs_base; |
@@ -1087,20 +1207,34 @@ static int samsung_i2s_probe(struct platform_device *pdev) | |||
1087 | sec_dai->dma_playback.dma_addr = regs_base + I2STXDS; | 1207 | sec_dai->dma_playback.dma_addr = regs_base + I2STXDS; |
1088 | sec_dai->dma_playback.client = | 1208 | sec_dai->dma_playback.client = |
1089 | (struct s3c2410_dma_client *)&sec_dai->dma_playback; | 1209 | (struct s3c2410_dma_client *)&sec_dai->dma_playback; |
1090 | /* Use iDMA always if SysDMA not provided */ | 1210 | sec_dai->dma_playback.ch_name = "tx-sec"; |
1091 | sec_dai->dma_playback.channel = dma_pl_sec_chan ? : -1; | 1211 | |
1212 | if (!np) { | ||
1213 | res = platform_get_resource(pdev, IORESOURCE_DMA, 2); | ||
1214 | if (res) | ||
1215 | sec_dai->dma_playback.channel = res->start; | ||
1216 | } | ||
1217 | |||
1092 | sec_dai->dma_playback.dma_size = 4; | 1218 | sec_dai->dma_playback.dma_size = 4; |
1093 | sec_dai->base = regs_base; | 1219 | sec_dai->base = regs_base; |
1094 | sec_dai->quirks = quirks; | 1220 | sec_dai->quirks = quirks; |
1095 | sec_dai->idma_playback.dma_addr = i2s_cfg->idma_addr; | 1221 | sec_dai->idma_playback.dma_addr = idma_addr; |
1096 | sec_dai->pri_dai = pri_dai; | 1222 | sec_dai->pri_dai = pri_dai; |
1097 | pri_dai->sec_dai = sec_dai; | 1223 | pri_dai->sec_dai = sec_dai; |
1098 | } | 1224 | } |
1099 | 1225 | ||
1100 | if (i2s_pdata->cfg_gpio && i2s_pdata->cfg_gpio(pdev)) { | 1226 | if (np) { |
1101 | dev_err(&pdev->dev, "Unable to configure gpio\n"); | 1227 | if (samsung_i2s_parse_dt_gpio(pri_dai)) { |
1102 | ret = -EINVAL; | 1228 | dev_err(&pdev->dev, "Unable to configure gpio\n"); |
1103 | goto err; | 1229 | ret = -EINVAL; |
1230 | goto err; | ||
1231 | } | ||
1232 | } else { | ||
1233 | if (i2s_pdata->cfg_gpio && i2s_pdata->cfg_gpio(pdev)) { | ||
1234 | dev_err(&pdev->dev, "Unable to configure gpio\n"); | ||
1235 | ret = -EINVAL; | ||
1236 | goto err; | ||
1237 | } | ||
1104 | } | 1238 | } |
1105 | 1239 | ||
1106 | snd_soc_register_dai(&pri_dai->pdev->dev, &pri_dai->i2s_dai_drv); | 1240 | snd_soc_register_dai(&pri_dai->pdev->dev, &pri_dai->i2s_dai_drv); |
@@ -1120,10 +1254,14 @@ static int samsung_i2s_remove(struct platform_device *pdev) | |||
1120 | { | 1254 | { |
1121 | struct i2s_dai *i2s, *other; | 1255 | struct i2s_dai *i2s, *other; |
1122 | struct resource *res; | 1256 | struct resource *res; |
1257 | struct s3c_audio_pdata *i2s_pdata = pdev->dev.platform_data; | ||
1123 | 1258 | ||
1124 | i2s = dev_get_drvdata(&pdev->dev); | 1259 | i2s = dev_get_drvdata(&pdev->dev); |
1125 | other = i2s->pri_dai ? : i2s->sec_dai; | 1260 | other = i2s->pri_dai ? : i2s->sec_dai; |
1126 | 1261 | ||
1262 | if (!i2s_pdata->cfg_gpio && pdev->dev.of_node) | ||
1263 | samsung_i2s_dt_gpio_free(i2s->pri_dai); | ||
1264 | |||
1127 | if (other) { | 1265 | if (other) { |
1128 | other->pri_dai = NULL; | 1266 | other->pri_dai = NULL; |
1129 | other->sec_dai = NULL; | 1267 | other->sec_dai = NULL; |
@@ -1143,12 +1281,47 @@ static int samsung_i2s_remove(struct platform_device *pdev) | |||
1143 | return 0; | 1281 | return 0; |
1144 | } | 1282 | } |
1145 | 1283 | ||
1284 | static struct platform_device_id samsung_i2s_driver_ids[] = { | ||
1285 | { | ||
1286 | .name = "samsung-i2s", | ||
1287 | .driver_data = TYPE_PRI, | ||
1288 | }, { | ||
1289 | .name = "samsung-i2s-sec", | ||
1290 | .driver_data = TYPE_SEC, | ||
1291 | }, | ||
1292 | {}, | ||
1293 | }; | ||
1294 | MODULE_DEVICE_TABLE(platform, samsung-i2s-driver-ids); | ||
1295 | |||
1296 | #ifdef CONFIG_OF | ||
1297 | static struct samsung_i2s_dai_data samsung_i2s_dai_data_array[] = { | ||
1298 | [TYPE_PRI] = { TYPE_PRI }, | ||
1299 | [TYPE_SEC] = { TYPE_SEC }, | ||
1300 | }; | ||
1301 | |||
1302 | static const struct of_device_id exynos_i2s_match[] = { | ||
1303 | { .compatible = "samsung,i2s-v5", | ||
1304 | .data = &samsung_i2s_dai_data_array[TYPE_PRI], | ||
1305 | }, | ||
1306 | {}, | ||
1307 | }; | ||
1308 | MODULE_DEVICE_TABLE(of, exynos_i2s_match); | ||
1309 | #endif | ||
1310 | |||
1311 | static const struct dev_pm_ops samsung_i2s_pm = { | ||
1312 | SET_RUNTIME_PM_OPS(i2s_runtime_suspend, | ||
1313 | i2s_runtime_resume, NULL) | ||
1314 | }; | ||
1315 | |||
1146 | static struct platform_driver samsung_i2s_driver = { | 1316 | static struct platform_driver samsung_i2s_driver = { |
1147 | .probe = samsung_i2s_probe, | 1317 | .probe = samsung_i2s_probe, |
1148 | .remove = samsung_i2s_remove, | 1318 | .remove = samsung_i2s_remove, |
1319 | .id_table = samsung_i2s_driver_ids, | ||
1149 | .driver = { | 1320 | .driver = { |
1150 | .name = "samsung-i2s", | 1321 | .name = "samsung-i2s", |
1151 | .owner = THIS_MODULE, | 1322 | .owner = THIS_MODULE, |
1323 | .of_match_table = of_match_ptr(exynos_i2s_match), | ||
1324 | .pm = &samsung_i2s_pm, | ||
1152 | }, | 1325 | }, |
1153 | }; | 1326 | }; |
1154 | 1327 | ||