aboutsummaryrefslogtreecommitdiffstats
path: root/sound/soc/soc-core.c
diff options
context:
space:
mode:
Diffstat (limited to 'sound/soc/soc-core.c')
-rw-r--r--sound/soc/soc-core.c66
1 files changed, 38 insertions, 28 deletions
diff --git a/sound/soc/soc-core.c b/sound/soc/soc-core.c
index bd656db347ee..e748b00466b7 100644
--- a/sound/soc/soc-core.c
+++ b/sound/soc/soc-core.c
@@ -1213,7 +1213,6 @@ struct snd_kcontrol *snd_soc_cnew(const struct snd_kcontrol_new *_template,
1213 memcpy(&template, _template, sizeof(template)); 1213 memcpy(&template, _template, sizeof(template));
1214 if (long_name) 1214 if (long_name)
1215 template.name = long_name; 1215 template.name = long_name;
1216 template.access = SNDRV_CTL_ELEM_ACCESS_READWRITE;
1217 template.index = 0; 1216 template.index = 0;
1218 1217
1219 return snd_ctl_new1(&template, data); 1218 return snd_ctl_new1(&template, data);
@@ -1348,13 +1347,16 @@ EXPORT_SYMBOL_GPL(snd_soc_info_enum_ext);
1348int snd_soc_info_volsw_ext(struct snd_kcontrol *kcontrol, 1347int snd_soc_info_volsw_ext(struct snd_kcontrol *kcontrol,
1349 struct snd_ctl_elem_info *uinfo) 1348 struct snd_ctl_elem_info *uinfo)
1350{ 1349{
1351 int mask = kcontrol->private_value; 1350 int max = kcontrol->private_value;
1351
1352 if (max == 1)
1353 uinfo->type = SNDRV_CTL_ELEM_TYPE_BOOLEAN;
1354 else
1355 uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;
1352 1356
1353 uinfo->type =
1354 mask == 1 ? SNDRV_CTL_ELEM_TYPE_BOOLEAN : SNDRV_CTL_ELEM_TYPE_INTEGER;
1355 uinfo->count = 1; 1357 uinfo->count = 1;
1356 uinfo->value.integer.min = 0; 1358 uinfo->value.integer.min = 0;
1357 uinfo->value.integer.max = mask; 1359 uinfo->value.integer.max = max;
1358 return 0; 1360 return 0;
1359} 1361}
1360EXPORT_SYMBOL_GPL(snd_soc_info_volsw_ext); 1362EXPORT_SYMBOL_GPL(snd_soc_info_volsw_ext);
@@ -1371,15 +1373,18 @@ EXPORT_SYMBOL_GPL(snd_soc_info_volsw_ext);
1371int snd_soc_info_volsw(struct snd_kcontrol *kcontrol, 1373int snd_soc_info_volsw(struct snd_kcontrol *kcontrol,
1372 struct snd_ctl_elem_info *uinfo) 1374 struct snd_ctl_elem_info *uinfo)
1373{ 1375{
1374 int mask = (kcontrol->private_value >> 16) & 0xff; 1376 int max = (kcontrol->private_value >> 16) & 0xff;
1375 int shift = (kcontrol->private_value >> 8) & 0x0f; 1377 int shift = (kcontrol->private_value >> 8) & 0x0f;
1376 int rshift = (kcontrol->private_value >> 12) & 0x0f; 1378 int rshift = (kcontrol->private_value >> 12) & 0x0f;
1377 1379
1378 uinfo->type = 1380 if (max == 1)
1379 mask == 1 ? SNDRV_CTL_ELEM_TYPE_BOOLEAN : SNDRV_CTL_ELEM_TYPE_INTEGER; 1381 uinfo->type = SNDRV_CTL_ELEM_TYPE_BOOLEAN;
1382 else
1383 uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;
1384
1380 uinfo->count = shift == rshift ? 1 : 2; 1385 uinfo->count = shift == rshift ? 1 : 2;
1381 uinfo->value.integer.min = 0; 1386 uinfo->value.integer.min = 0;
1382 uinfo->value.integer.max = mask; 1387 uinfo->value.integer.max = max;
1383 return 0; 1388 return 0;
1384} 1389}
1385EXPORT_SYMBOL_GPL(snd_soc_info_volsw); 1390EXPORT_SYMBOL_GPL(snd_soc_info_volsw);
@@ -1400,7 +1405,8 @@ int snd_soc_get_volsw(struct snd_kcontrol *kcontrol,
1400 int reg = kcontrol->private_value & 0xff; 1405 int reg = kcontrol->private_value & 0xff;
1401 int shift = (kcontrol->private_value >> 8) & 0x0f; 1406 int shift = (kcontrol->private_value >> 8) & 0x0f;
1402 int rshift = (kcontrol->private_value >> 12) & 0x0f; 1407 int rshift = (kcontrol->private_value >> 12) & 0x0f;
1403 int mask = (kcontrol->private_value >> 16) & 0xff; 1408 int max = (kcontrol->private_value >> 16) & 0xff;
1409 int mask = (1 << fls(max)) - 1;
1404 int invert = (kcontrol->private_value >> 24) & 0x01; 1410 int invert = (kcontrol->private_value >> 24) & 0x01;
1405 1411
1406 ucontrol->value.integer.value[0] = 1412 ucontrol->value.integer.value[0] =
@@ -1410,10 +1416,10 @@ int snd_soc_get_volsw(struct snd_kcontrol *kcontrol,
1410 (snd_soc_read(codec, reg) >> rshift) & mask; 1416 (snd_soc_read(codec, reg) >> rshift) & mask;
1411 if (invert) { 1417 if (invert) {
1412 ucontrol->value.integer.value[0] = 1418 ucontrol->value.integer.value[0] =
1413 mask - ucontrol->value.integer.value[0]; 1419 max - ucontrol->value.integer.value[0];
1414 if (shift != rshift) 1420 if (shift != rshift)
1415 ucontrol->value.integer.value[1] = 1421 ucontrol->value.integer.value[1] =
1416 mask - ucontrol->value.integer.value[1]; 1422 max - ucontrol->value.integer.value[1];
1417 } 1423 }
1418 1424
1419 return 0; 1425 return 0;
@@ -1436,25 +1442,24 @@ int snd_soc_put_volsw(struct snd_kcontrol *kcontrol,
1436 int reg = kcontrol->private_value & 0xff; 1442 int reg = kcontrol->private_value & 0xff;
1437 int shift = (kcontrol->private_value >> 8) & 0x0f; 1443 int shift = (kcontrol->private_value >> 8) & 0x0f;
1438 int rshift = (kcontrol->private_value >> 12) & 0x0f; 1444 int rshift = (kcontrol->private_value >> 12) & 0x0f;
1439 int mask = (kcontrol->private_value >> 16) & 0xff; 1445 int max = (kcontrol->private_value >> 16) & 0xff;
1446 int mask = (1 << fls(max)) - 1;
1440 int invert = (kcontrol->private_value >> 24) & 0x01; 1447 int invert = (kcontrol->private_value >> 24) & 0x01;
1441 int err;
1442 unsigned short val, val2, val_mask; 1448 unsigned short val, val2, val_mask;
1443 1449
1444 val = (ucontrol->value.integer.value[0] & mask); 1450 val = (ucontrol->value.integer.value[0] & mask);
1445 if (invert) 1451 if (invert)
1446 val = mask - val; 1452 val = max - val;
1447 val_mask = mask << shift; 1453 val_mask = mask << shift;
1448 val = val << shift; 1454 val = val << shift;
1449 if (shift != rshift) { 1455 if (shift != rshift) {
1450 val2 = (ucontrol->value.integer.value[1] & mask); 1456 val2 = (ucontrol->value.integer.value[1] & mask);
1451 if (invert) 1457 if (invert)
1452 val2 = mask - val2; 1458 val2 = max - val2;
1453 val_mask |= mask << rshift; 1459 val_mask |= mask << rshift;
1454 val |= val2 << rshift; 1460 val |= val2 << rshift;
1455 } 1461 }
1456 err = snd_soc_update_bits(codec, reg, val_mask, val); 1462 return snd_soc_update_bits(codec, reg, val_mask, val);
1457 return err;
1458} 1463}
1459EXPORT_SYMBOL_GPL(snd_soc_put_volsw); 1464EXPORT_SYMBOL_GPL(snd_soc_put_volsw);
1460 1465
@@ -1471,13 +1476,16 @@ EXPORT_SYMBOL_GPL(snd_soc_put_volsw);
1471int snd_soc_info_volsw_2r(struct snd_kcontrol *kcontrol, 1476int snd_soc_info_volsw_2r(struct snd_kcontrol *kcontrol,
1472 struct snd_ctl_elem_info *uinfo) 1477 struct snd_ctl_elem_info *uinfo)
1473{ 1478{
1474 int mask = (kcontrol->private_value >> 12) & 0xff; 1479 int max = (kcontrol->private_value >> 12) & 0xff;
1480
1481 if (max == 1)
1482 uinfo->type = SNDRV_CTL_ELEM_TYPE_BOOLEAN;
1483 else
1484 uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;
1475 1485
1476 uinfo->type =
1477 mask == 1 ? SNDRV_CTL_ELEM_TYPE_BOOLEAN : SNDRV_CTL_ELEM_TYPE_INTEGER;
1478 uinfo->count = 2; 1486 uinfo->count = 2;
1479 uinfo->value.integer.min = 0; 1487 uinfo->value.integer.min = 0;
1480 uinfo->value.integer.max = mask; 1488 uinfo->value.integer.max = max;
1481 return 0; 1489 return 0;
1482} 1490}
1483EXPORT_SYMBOL_GPL(snd_soc_info_volsw_2r); 1491EXPORT_SYMBOL_GPL(snd_soc_info_volsw_2r);
@@ -1498,7 +1506,8 @@ int snd_soc_get_volsw_2r(struct snd_kcontrol *kcontrol,
1498 int reg = kcontrol->private_value & 0xff; 1506 int reg = kcontrol->private_value & 0xff;
1499 int reg2 = (kcontrol->private_value >> 24) & 0xff; 1507 int reg2 = (kcontrol->private_value >> 24) & 0xff;
1500 int shift = (kcontrol->private_value >> 8) & 0x0f; 1508 int shift = (kcontrol->private_value >> 8) & 0x0f;
1501 int mask = (kcontrol->private_value >> 12) & 0xff; 1509 int max = (kcontrol->private_value >> 12) & 0xff;
1510 int mask = (1<<fls(max))-1;
1502 int invert = (kcontrol->private_value >> 20) & 0x01; 1511 int invert = (kcontrol->private_value >> 20) & 0x01;
1503 1512
1504 ucontrol->value.integer.value[0] = 1513 ucontrol->value.integer.value[0] =
@@ -1507,9 +1516,9 @@ int snd_soc_get_volsw_2r(struct snd_kcontrol *kcontrol,
1507 (snd_soc_read(codec, reg2) >> shift) & mask; 1516 (snd_soc_read(codec, reg2) >> shift) & mask;
1508 if (invert) { 1517 if (invert) {
1509 ucontrol->value.integer.value[0] = 1518 ucontrol->value.integer.value[0] =
1510 mask - ucontrol->value.integer.value[0]; 1519 max - ucontrol->value.integer.value[0];
1511 ucontrol->value.integer.value[1] = 1520 ucontrol->value.integer.value[1] =
1512 mask - ucontrol->value.integer.value[1]; 1521 max - ucontrol->value.integer.value[1];
1513 } 1522 }
1514 1523
1515 return 0; 1524 return 0;
@@ -1532,7 +1541,8 @@ int snd_soc_put_volsw_2r(struct snd_kcontrol *kcontrol,
1532 int reg = kcontrol->private_value & 0xff; 1541 int reg = kcontrol->private_value & 0xff;
1533 int reg2 = (kcontrol->private_value >> 24) & 0xff; 1542 int reg2 = (kcontrol->private_value >> 24) & 0xff;
1534 int shift = (kcontrol->private_value >> 8) & 0x0f; 1543 int shift = (kcontrol->private_value >> 8) & 0x0f;
1535 int mask = (kcontrol->private_value >> 12) & 0xff; 1544 int max = (kcontrol->private_value >> 12) & 0xff;
1545 int mask = (1 << fls(max)) - 1;
1536 int invert = (kcontrol->private_value >> 20) & 0x01; 1546 int invert = (kcontrol->private_value >> 20) & 0x01;
1537 int err; 1547 int err;
1538 unsigned short val, val2, val_mask; 1548 unsigned short val, val2, val_mask;
@@ -1542,8 +1552,8 @@ int snd_soc_put_volsw_2r(struct snd_kcontrol *kcontrol,
1542 val2 = (ucontrol->value.integer.value[1] & mask); 1552 val2 = (ucontrol->value.integer.value[1] & mask);
1543 1553
1544 if (invert) { 1554 if (invert) {
1545 val = mask - val; 1555 val = max - val;
1546 val2 = mask - val2; 1556 val2 = max - val2;
1547 } 1557 }
1548 1558
1549 val = val << shift; 1559 val = val << shift;