Changeset 694
- Timestamp:
- Aug 15, 2021, 1:07:08 AM (4 years ago)
- Location:
- GPL/branches/uniaud32-next
- Files:
-
- 37 edited
Legend:
- Unmodified
- Added
- Removed
-
TabularUnified GPL/branches/uniaud32-next/alsa-kernel/core/control.c ¶
r693 r694 8 8 #include <linux/interrupt.h> 9 9 #include <linux/module.h> 10 #include <linux/moduleparam.h> 10 11 #include <linux/slab.h> 11 12 #include <linux/vmalloc.h> … … 19 20 #include <sound/control.h> 20 21 21 /* max number of user-defined controls */ 22 #define MAX_USER_CONTROLS 32 22 // Max allocation size for user controls. 23 static int max_user_ctl_alloc_size = 8 * 1024 * 1024; 24 module_param_named(max_user_ctl_alloc_size, max_user_ctl_alloc_size, int, 0444); 25 MODULE_PARM_DESC(max_user_ctl_alloc_size, "Max allocation size for user controls"); 26 23 27 #define MAX_CONTROL_COUNT 1028 24 28 … … 31 35 32 36 static DECLARE_RWSEM(snd_ioctl_rwsem); 37 static DECLARE_RWSEM(snd_ctl_layer_rwsem); 33 38 static LIST_HEAD(snd_control_ioctls); 34 39 #ifdef CONFIG_COMPAT 35 40 static LIST_HEAD(snd_control_compat_ioctls); 36 41 #endif 42 static struct snd_ctl_layer_ops *snd_ctl_layer; 37 43 38 44 static int snd_ctl_open(struct inode *inode, struct file *file) … … 185 191 186 192 /** 193 * snd_ctl_notify_one - Send notification to user-space for a control change 194 * @card: the card to send notification 195 * @mask: the event mask, SNDRV_CTL_EVENT_* 196 * @kctl: the pointer with the control instance 197 * @ioff: the additional offset to the control index 198 * 199 * This function calls snd_ctl_notify() and does additional jobs 200 * like LED state changes. 201 */ 202 void snd_ctl_notify_one(struct snd_card *card, unsigned int mask, 203 struct snd_kcontrol *kctl, unsigned int ioff) 204 { 205 struct snd_ctl_elem_id id = kctl->id; 206 struct snd_ctl_layer_ops *lops; 207 208 id.index += ioff; 209 id.numid += ioff; 210 snd_ctl_notify(card, mask, &id); 211 down_read(&snd_ctl_layer_rwsem); 212 for (lops = snd_ctl_layer; lops; lops = lops->next) 213 lops->lnotify(card, mask, kctl, ioff); 214 up_read(&snd_ctl_layer_rwsem); 215 } 216 EXPORT_SYMBOL(snd_ctl_notify_one); 217 218 /** 187 219 * snd_ctl_new - create a new control instance with some elements 188 220 * @kctl: the pointer to store new control instance … … 263 295 SNDRV_CTL_ELEM_ACCESS_TLV_COMMAND | 264 296 SNDRV_CTL_ELEM_ACCESS_TLV_CALLBACK | 297 SNDRV_CTL_ELEM_ACCESS_LED_MASK | 265 298 SNDRV_CTL_ELEM_ACCESS_SKIP_CHECK); 266 299 … … 355 388 struct snd_ctl_elem_id id; 356 389 unsigned int idx; 357 unsigned int count;358 390 struct snd_kcontrol *old; 359 391 int err; … … 389 421 card->last_numid += kcontrol->count; 390 422 391 id = kcontrol->id; 392 count = kcontrol->count; 393 for (idx = 0; idx < count; idx++, id.index++, id.numid++) 394 snd_ctl_notify(card, SNDRV_CTL_EVENT_MASK_ADD, &id); 423 for (idx = 0; idx < kcontrol->count; idx++) 424 snd_ctl_notify_one(card, SNDRV_CTL_EVENT_MASK_ADD, kcontrol, idx); 395 425 396 426 return 0; … … 475 505 int snd_ctl_remove(struct snd_card *card, struct snd_kcontrol *kcontrol) 476 506 { 477 struct snd_ctl_elem_id id;478 507 unsigned int idx; 479 508 … … 482 511 list_del(&kcontrol->list); 483 512 card->controls_count -= kcontrol->count; 484 id = kcontrol->id; 485 for (idx = 0; idx < kcontrol->count; idx++, id.index++, id.numid++) 486 snd_ctl_notify(card, SNDRV_CTL_EVENT_MASK_REMOVE, &id); 513 for (idx = 0; idx < kcontrol->count; idx++) 514 snd_ctl_notify_one(card, SNDRV_CTL_EVENT_MASK_REMOVE, kcontrol, idx); 487 515 snd_ctl_free_one(kcontrol); 488 516 return 0; … … 550 578 } 551 579 ret = snd_ctl_remove(card, kctl); 552 if (ret < 0)553 goto error;554 card->user_ctl_count--;555 580 error: 556 581 up_write(&card->controls_rwsem); … … 597 622 } 598 623 snd_ctl_build_ioff(id, kctl, index_offset); 599 ret = 1; 624 downgrade_write(&card->controls_rwsem); 625 snd_ctl_notify_one(card, SNDRV_CTL_EVENT_MASK_INFO, kctl, index_offset); 626 up_read(&card->controls_rwsem); 627 return 1; 628 600 629 unlock: 601 630 up_write(&card->controls_rwsem); 602 if (ret > 0)603 snd_ctl_notify(card, SNDRV_CTL_EVENT_MASK_INFO, id);604 631 return ret; 605 632 } … … 1035 1062 return result; 1036 1063 /* drop internal access flags */ 1037 info.access &= ~SNDRV_CTL_ELEM_ACCESS_SKIP_CHECK; 1064 info.access &= ~(SNDRV_CTL_ELEM_ACCESS_SKIP_CHECK| 1065 SNDRV_CTL_ELEM_ACCESS_LED_MASK); 1038 1066 if (copy_to_user(_info, &info, sizeof(info))) 1039 1067 return -EFAULT; … … 1123 1151 int result; 1124 1152 1153 down_write(&card->controls_rwsem); 1125 1154 kctl = snd_ctl_find_id(card, &control->id); 1126 if (kctl == NULL) 1155 if (kctl == NULL) { 1156 up_write(&card->controls_rwsem); 1127 1157 return -ENOENT; 1158 } 1128 1159 1129 1160 index_offset = snd_ctl_get_ioff(kctl, &control->id); … … 1131 1162 if (!(vd->access & SNDRV_CTL_ELEM_ACCESS_WRITE) || kctl->put == NULL || 1132 1163 (file && vd->owner && vd->owner != file)) { 1164 up_write(&card->controls_rwsem); 1133 1165 return -EPERM; 1134 1166 } … … 1136 1168 snd_ctl_build_ioff(&control->id, kctl, index_offset); 1137 1169 result = kctl->put(kctl, control); 1138 if (result < 0) 1170 if (result < 0) { 1171 up_write(&card->controls_rwsem); 1139 1172 return result; 1173 } 1140 1174 1141 1175 if (result > 0) { 1142 struct snd_ctl_elem_id id = control->id; 1143 snd_ctl_notify(card, SNDRV_CTL_EVENT_MASK_VALUE, &id); 1176 downgrade_write(&card->controls_rwsem); 1177 snd_ctl_notify_one(card, SNDRV_CTL_EVENT_MASK_VALUE, kctl, index_offset); 1178 up_read(&card->controls_rwsem); 1179 } else { 1180 up_write(&card->controls_rwsem); 1144 1181 } 1145 1182 … … 1163 1200 goto error; 1164 1201 1165 down_write(&card->controls_rwsem);1166 1202 result = snd_ctl_elem_write(card, file, control); 1167 up_write(&card->controls_rwsem);1168 1203 if (result < 0) 1169 1204 goto error; … … 1244 1279 }; 1245 1280 1281 // check whether the addition (in bytes) of user ctl element may overflow the limit. 1282 static bool check_user_elem_overflow(struct snd_card *card, ssize_t add) 1283 { 1284 return (ssize_t)card->user_ctl_alloc_size + add > max_user_ctl_alloc_size; 1285 } 1286 1246 1287 static int snd_ctl_elem_user_info(struct snd_kcontrol *kcontrol, 1247 1288 struct snd_ctl_elem_info *uinfo) … … 1309 1350 } 1310 1351 1352 /* called in controls_rwsem write lock */ 1311 1353 static int replace_user_tlv(struct snd_kcontrol *kctl, unsigned int __user *buf, 1312 1354 unsigned int size) … … 1314 1356 struct user_element *ue = kctl->private_data; 1315 1357 unsigned int *container; 1316 struct snd_ctl_elem_id id;1317 1358 unsigned int mask = 0; 1318 1359 int i; … … 1321 1362 if (size > 1024 * 128) /* sane value */ 1322 1363 return -EINVAL; 1364 1365 // does the TLV size change cause overflow? 1366 if (check_user_elem_overflow(ue->card, (ssize_t)(size - ue->tlv_data_size))) 1367 return -ENOMEM; 1323 1368 1324 1369 container = vmemdup_user(buf, size); … … 1339 1384 kctl->vd[i].access |= SNDRV_CTL_ELEM_ACCESS_TLV_READ; 1340 1385 mask = SNDRV_CTL_EVENT_MASK_INFO; 1341 } 1342 1343 kvfree(ue->tlv_data); 1386 } else { 1387 ue->card->user_ctl_alloc_size -= ue->tlv_data_size; 1388 ue->tlv_data_size = 0; 1389 kvfree(ue->tlv_data); 1390 } 1391 1344 1392 ue->tlv_data = container; 1345 1393 ue->tlv_data_size = size; 1394 // decremented at private_free. 1395 ue->card->user_ctl_alloc_size += size; 1346 1396 1347 1397 mask |= SNDRV_CTL_EVENT_MASK_TLV; 1348 for (i = 0; i < kctl->count; ++i) { 1349 snd_ctl_build_ioff(&id, kctl, i); 1350 snd_ctl_notify(ue->card, mask, &id); 1351 } 1398 for (i = 0; i < kctl->count; ++i) 1399 snd_ctl_notify_one(ue->card, mask, kctl, i); 1352 1400 1353 1401 return change; … … 1380 1428 } 1381 1429 1430 /* called in controls_rwsem write lock */ 1382 1431 static int snd_ctl_elem_init_enum_names(struct user_element *ue) 1383 1432 { … … 1388 1437 uintptr_t user_ptrval = ue->info.value.enumerated.names_ptr; 1389 1438 1390 if (ue->info.value.enumerated.names_length > 64 * 1024) 1439 buf_len = ue->info.value.enumerated.names_length; 1440 if (buf_len > 64 * 1024) 1391 1441 return -EINVAL; 1392 1442 1393 names = vmemdup_user((const void __user *)user_ptrval, 1394 ue->info.value.enumerated.names_length); 1443 if (check_user_elem_overflow(ue->card, buf_len)) 1444 return -ENOMEM; 1445 names = vmemdup_user((const void __user *)user_ptrval, buf_len); 1395 1446 if (IS_ERR(names)) 1396 1447 return PTR_ERR(names); 1397 1448 1398 1449 /* check that there are enough valid names */ 1399 buf_len = ue->info.value.enumerated.names_length;1400 1450 p = names; 1401 1451 for (i = 0; i < ue->info.value.enumerated.items; ++i) { … … 1411 1461 ue->priv_data = names; 1412 1462 ue->info.value.enumerated.names_ptr = 0; 1413 1414 return 0; 1463 // increment the allocation size; decremented again at private_free. 1464 ue->card->user_ctl_alloc_size += ue->info.value.enumerated.names_length; 1465 1466 return 0; 1467 } 1468 1469 static size_t compute_user_elem_size(size_t size, unsigned int count) 1470 { 1471 return sizeof(struct user_element) + size * count; 1415 1472 } 1416 1473 … … 1418 1475 { 1419 1476 struct user_element *ue = kcontrol->private_data; 1477 1478 // decrement the allocation size. 1479 ue->card->user_ctl_alloc_size -= compute_user_elem_size(ue->elem_data_size, kcontrol->count); 1480 ue->card->user_ctl_alloc_size -= ue->tlv_data_size; 1481 if (ue->priv_data) 1482 ue->card->user_ctl_alloc_size -= ue->info.value.enumerated.names_length; 1420 1483 1421 1484 kvfree(ue->tlv_data); … … 1432 1495 unsigned int access; 1433 1496 long private_size; 1497 size_t alloc_size; 1434 1498 struct user_element *ue; 1435 1499 unsigned int offset; … … 1448 1512 return err; 1449 1513 } 1450 1451 /*1452 * The number of userspace controls are counted control by control,1453 * not element by element.1454 */1455 if (card->user_ctl_count + 1 > MAX_USER_CONTROLS)1456 return -ENOMEM;1457 1514 1458 1515 /* Check the number of elements for this userspace control. */ … … 1486 1543 return -EINVAL; 1487 1544 private_size = value_sizes[info->type] * info->count; 1545 alloc_size = compute_user_elem_size(private_size, count); 1546 1547 down_write(&card->controls_rwsem); 1548 if (check_user_elem_overflow(card, alloc_size)) { 1549 err = -ENOMEM; 1550 goto unlock; 1551 } 1488 1552 1489 1553 /* … … 1495 1559 err = snd_ctl_new(&kctl, count, access, file); 1496 1560 if (err < 0) 1497 return err;1561 goto unlock; 1498 1562 memcpy(&kctl->id, &info->id, sizeof(kctl->id)); 1499 kctl->private_data = kzalloc(sizeof(struct user_element) + private_size * count, 1500 GFP_KERNEL); 1501 if (kctl->private_data == NULL) { 1563 ue = kzalloc(alloc_size, GFP_KERNEL); 1564 if (!ue) { 1502 1565 kfree(kctl); 1503 return -ENOMEM; 1504 } 1566 err = -ENOMEM; 1567 goto unlock; 1568 } 1569 kctl->private_data = ue; 1505 1570 kctl->private_free = snd_ctl_elem_user_free; 1506 1571 1572 // increment the allocated size; decremented again at private_free. 1573 card->user_ctl_alloc_size += alloc_size; 1574 1507 1575 /* Set private data for this userspace control. */ 1508 ue = (struct user_element *)kctl->private_data;1509 1576 ue->card = card; 1510 1577 ue->info = *info; … … 1516 1583 if (err < 0) { 1517 1584 snd_ctl_free_one(kctl); 1518 return err;1585 goto unlock; 1519 1586 } 1520 1587 } … … 1533 1600 1534 1601 /* This function manage to free the instance on failure. */ 1535 down_write(&card->controls_rwsem);1536 1602 err = __snd_ctl_add_replace(card, kctl, CTL_ADD_EXCLUSIVE); 1537 1603 if (err < 0) { … … 1548 1614 * which locks the element. 1549 1615 */ 1550 1551 card->user_ctl_count++;1552 1553 1616 unlock: 1554 1617 up_write(&card->controls_rwsem); … … 1990 2053 1991 2054 /* 2055 * control layers (audio LED etc.) 2056 */ 2057 2058 /** 2059 * snd_ctl_request_layer - request to use the layer 2060 * @module_name: Name of the kernel module (NULL == build-in) 2061 * 2062 * Return an error code when the module cannot be loaded. 2063 */ 2064 int snd_ctl_request_layer(const char *module_name) 2065 { 2066 struct snd_ctl_layer_ops *lops; 2067 2068 if (module_name == NULL) 2069 return 0; 2070 down_read(&snd_ctl_layer_rwsem); 2071 for (lops = snd_ctl_layer; lops; lops = lops->next) 2072 if (strcmp(lops->module_name, module_name) == 0) 2073 break; 2074 up_read(&snd_ctl_layer_rwsem); 2075 if (lops) 2076 return 0; 2077 return request_module(module_name); 2078 } 2079 EXPORT_SYMBOL_GPL(snd_ctl_request_layer); 2080 2081 /** 2082 * snd_ctl_register_layer - register new control layer 2083 * @lops: operation structure 2084 * 2085 * The new layer can track all control elements and do additional 2086 * operations on top (like audio LED handling). 2087 */ 2088 void snd_ctl_register_layer(struct snd_ctl_layer_ops *lops) 2089 { 2090 struct snd_card *card; 2091 int card_number; 2092 2093 down_write(&snd_ctl_layer_rwsem); 2094 lops->next = snd_ctl_layer; 2095 snd_ctl_layer = lops; 2096 up_write(&snd_ctl_layer_rwsem); 2097 for (card_number = 0; card_number < SNDRV_CARDS; card_number++) { 2098 card = snd_card_ref(card_number); 2099 if (card) { 2100 down_read(&card->controls_rwsem); 2101 lops->lregister(card); 2102 up_read(&card->controls_rwsem); 2103 snd_card_unref(card); 2104 } 2105 } 2106 } 2107 EXPORT_SYMBOL_GPL(snd_ctl_register_layer); 2108 2109 /** 2110 * snd_ctl_disconnect_layer - disconnect control layer 2111 * @lops: operation structure 2112 * 2113 * It is expected that the information about tracked cards 2114 * is freed before this call (the disconnect callback is 2115 * not called here). 2116 */ 2117 void snd_ctl_disconnect_layer(struct snd_ctl_layer_ops *lops) 2118 { 2119 struct snd_ctl_layer_ops *lops2, *prev_lops2; 2120 2121 down_write(&snd_ctl_layer_rwsem); 2122 for (lops2 = snd_ctl_layer, prev_lops2 = NULL; lops2; lops2 = lops2->next) { 2123 if (lops2 == lops) { 2124 if (!prev_lops2) 2125 snd_ctl_layer = lops->next; 2126 else 2127 prev_lops2->next = lops->next; 2128 break; 2129 } 2130 prev_lops2 = lops2; 2131 } 2132 up_write(&snd_ctl_layer_rwsem); 2133 } 2134 EXPORT_SYMBOL_GPL(snd_ctl_disconnect_layer); 2135 2136 /* 1992 2137 * INIT PART 1993 2138 */ … … 2012 2157 { 2013 2158 struct snd_card *card = device->device_data; 2014 2015 return snd_register_device(SNDRV_DEVICE_TYPE_CONTROL, card, -1, 2016 &snd_ctl_f_ops, card, &card->ctl_dev); 2159 struct snd_ctl_layer_ops *lops; 2160 int err; 2161 2162 err = snd_register_device(SNDRV_DEVICE_TYPE_CONTROL, card, -1, 2163 &snd_ctl_f_ops, card, &card->ctl_dev); 2164 if (err < 0) 2165 return err; 2166 down_read(&card->controls_rwsem); 2167 down_read(&snd_ctl_layer_rwsem); 2168 for (lops = snd_ctl_layer; lops; lops = lops->next) 2169 lops->lregister(card); 2170 up_read(&snd_ctl_layer_rwsem); 2171 up_read(&card->controls_rwsem); 2172 return 0; 2017 2173 } 2018 2174 … … 2024 2180 struct snd_card *card = device->device_data; 2025 2181 struct snd_ctl_file *ctl; 2182 struct snd_ctl_layer_ops *lops; 2026 2183 unsigned long flags; 2027 2184 … … 2032 2189 } 2033 2190 read_unlock_irqrestore(&card->ctl_files_rwlock, flags); 2191 2192 down_read(&card->controls_rwsem); 2193 down_read(&snd_ctl_layer_rwsem); 2194 for (lops = snd_ctl_layer; lops; lops = lops->next) 2195 lops->ldisconnect(card); 2196 up_read(&snd_ctl_layer_rwsem); 2197 up_read(&card->controls_rwsem); 2034 2198 2035 2199 return snd_unregister_device(&card->ctl_dev); -
TabularUnified GPL/branches/uniaud32-next/alsa-kernel/core/oss/pcm_oss.c ¶
r660 r694 3085 3085 } 3086 3086 #else /* !CONFIG_SND_VERBOSE_PROCFS */ 3087 #define snd_pcm_oss_proc_init(pcm) 3088 #define snd_pcm_oss_proc_done(pcm) 3087 static inline void snd_pcm_oss_proc_init(struct snd_pcm *pcm) 3088 { 3089 } 3090 static inline void snd_pcm_oss_proc_done(struct snd_pcm *pcm) 3091 { 3092 } 3089 3093 #endif /* CONFIG_SND_VERBOSE_PROCFS */ 3090 3094 -
TabularUnified GPL/branches/uniaud32-next/alsa-kernel/core/pcm_memory.c ¶
r693 r694 177 177 size, &new_dmab) < 0) { 178 178 buffer->error = -ENOMEM; 179 pr_debug("ALSA pcmC%dD%d%c,%d:%s: cannot preallocate for size %zu\n", 180 substream->pcm->card->number, substream->pcm->device, 181 substream->stream ? 'c' : 'p', substream->number, 182 substream->pcm->name, size); 179 183 return; 180 184 } … … 211 215 212 216 #else /* !CONFIG_SND_VERBOSE_PROCFS */ 213 #define preallocate_info_init(s) 217 static inline void preallocate_info_init(struct snd_pcm_substream *substream) 218 { 219 } 214 220 #endif /* CONFIG_SND_VERBOSE_PROCFS */ 215 221 … … 401 407 size, dmab) < 0) { 402 408 kfree(dmab); 409 pr_debug("ALSA pcmC%dD%d%c,%d:%s: cannot preallocate for size %zu\n", 410 substream->pcm->card->number, substream->pcm->device, 411 substream->stream ? 'c' : 'p', substream->number, 412 substream->pcm->name, size); 403 413 return -ENOMEM; 404 414 } -
TabularUnified GPL/branches/uniaud32-next/alsa-kernel/core/pcm_native.c ¶
r693 r694 251 251 return false; 252 252 253 if (substream->ops->mmap || 254 (substream->dma_buffer.dev.type != SNDRV_DMA_TYPE_DEV && 255 substream->dma_buffer.dev.type != SNDRV_DMA_TYPE_DEV_UC)) 253 if (substream->ops->mmap || substream->ops->page) 256 254 return true; 257 255 258 return dma_can_mmap(substream->dma_buffer.dev.dev); 256 switch (substream->dma_buffer.dev.type) { 257 case SNDRV_DMA_TYPE_UNKNOWN: 258 return false; 259 case SNDRV_DMA_TYPE_CONTINUOUS: 260 case SNDRV_DMA_TYPE_VMALLOC: 261 return true; 262 default: 263 return dma_can_mmap(substream->dma_buffer.dev.dev); 264 } 259 265 } 260 266 … … 1451 1457 substream->runtime->stop_operating = true; 1452 1458 } 1453 return 0; /* uncondit onally stop all substreams */1459 return 0; /* unconditionally stop all substreams */ 1454 1460 } 1455 1461 … … 1495 1501 * Unlike snd_pcm_stop(), this affects only the given stream. 1496 1502 * 1497 * Return: Zero if succes ful, or a negative error code.1503 * Return: Zero if successful, or a negative error code. 1498 1504 */ 1499 1505 int snd_pcm_drain_done(struct snd_pcm_substream *substream) … … 3101 3107 snd_pcm_stream_lock_irq(substream); 3102 3108 /* FIXME: we should consider the boundary for the sync from app */ 3103 if (!(sflags & SNDRV_PCM_SYNC_PTR_APPL)) 3104 control->appl_ptr = scontrol.appl_ptr; 3105 else 3109 if (!(sflags & SNDRV_PCM_SYNC_PTR_APPL)) { 3110 err = pcm_lib_apply_appl_ptr(substream, 3111 scontrol.appl_ptr); 3112 if (err < 0) { 3113 snd_pcm_stream_unlock_irq(substream); 3114 return err; 3115 } 3116 } else 3106 3117 scontrol.appl_ptr = control->appl_ptr % boundary; 3107 3118 if (!(sflags & SNDRV_PCM_SYNC_PTR_AVAIL_MIN)) -
TabularUnified GPL/branches/uniaud32-next/alsa-kernel/core/seq/seq_ports.c ¶
r693 r694 515 515 } 516 516 517 static void delete_and_unsubscribe_port(struct snd_seq_client *client, 518 struct snd_seq_client_port *port, 519 struct snd_seq_subscribers *subs, 520 bool is_src, bool ack) 517 /* called with grp->list_mutex held */ 518 static void __delete_and_unsubscribe_port(struct snd_seq_client *client, 519 struct snd_seq_client_port *port, 520 struct snd_seq_subscribers *subs, 521 bool is_src, bool ack) 521 522 { 522 523 struct snd_seq_port_subs_info *grp; … … 526 527 grp = is_src ? &port->c_src : &port->c_dest; 527 528 list = is_src ? &subs->src_list : &subs->dest_list; 528 down_write(&grp->list_mutex);529 529 write_lock_irq(&grp->list_lock); 530 530 empty = list_empty(list); … … 536 536 if (!empty) 537 537 unsubscribe_port(client, port, grp, &subs->info, ack); 538 } 539 540 static void delete_and_unsubscribe_port(struct snd_seq_client *client, 541 struct snd_seq_client_port *port, 542 struct snd_seq_subscribers *subs, 543 bool is_src, bool ack) 544 { 545 struct snd_seq_port_subs_info *grp; 546 547 grp = is_src ? &port->c_src : &port->c_dest; 548 down_write(&grp->list_mutex); 549 __delete_and_unsubscribe_port(client, port, subs, is_src, ack); 538 550 up_write(&grp->list_mutex); 539 551 } … … 591 603 struct snd_seq_port_subscribe *info) 592 604 { 593 struct snd_seq_port_subs_info * src = &src_port->c_src;605 struct snd_seq_port_subs_info *dest = &dest_port->c_dest; 594 606 struct snd_seq_subscribers *subs; 595 607 int err = -ENOENT; 596 608 597 down_write(&src->list_mutex); 609 /* always start from deleting the dest port for avoiding concurrent 610 * deletions 611 */ 612 down_write(&dest->list_mutex); 598 613 /* look for the connection */ 599 list_for_each_entry(subs, & src->list_head, src_list, struct snd_seq_subscribers) {614 list_for_each_entry(subs, &dest->list_head, dest_list, struct snd_seq_subscribers) { 600 615 if (match_subs_info(info, &subs->info)) { 601 atomic_dec(&subs->ref_count); /* mark as not ready */ 616 __delete_and_unsubscribe_port(dest_client, dest_port, 617 subs, false, 618 connector->number != dest_client->number); 602 619 err = 0; 603 620 break; 604 621 } 605 622 } 606 up_write(& src->list_mutex);623 up_write(&dest->list_mutex); 607 624 if (err < 0) 608 625 return err; … … 610 627 delete_and_unsubscribe_port(src_client, src_port, subs, true, 611 628 connector->number != src_client->number); 612 delete_and_unsubscribe_port(dest_client, dest_port, subs, false,613 connector->number != dest_client->number);614 629 kfree(subs); 615 630 return 0; -
TabularUnified GPL/branches/uniaud32-next/alsa-kernel/core/seq_device.c ¶
r629 r694 134 134 } 135 135 EXPORT_SYMBOL(snd_seq_device_load_drivers); 136 #define cancel_autoload_drivers() cancel_work_sync(&autoload_work) 136 137 static inline void cancel_autoload_drivers(void) 138 { 139 cancel_work_sync(&autoload_work); 140 } 137 141 #else 138 #define queue_autoload_drivers() /* NOP */ 139 #define cancel_autoload_drivers() /* NOP */ 142 static inline void queue_autoload_drivers(void) 143 { 144 } 145 146 static inline void cancel_autoload_drivers(void) 147 { 148 } 140 149 #endif 141 150 -
TabularUnified GPL/branches/uniaud32-next/alsa-kernel/hda/hdac_stream.c ¶
r693 r694 624 624 625 625 /** 626 * snd_hdac_stream_sync - sync with start/st rop trigger operation626 * snd_hdac_stream_sync - sync with start/stop trigger operation 627 627 * @azx_dev: HD-audio core stream (master stream) 628 628 * @start: true = start, false = stop -
TabularUnified GPL/branches/uniaud32-next/alsa-kernel/include/sound/control.h ¶
r629 r694 25 25 /* internal flag for skipping validations */ 26 26 #ifdef CONFIG_SND_CTL_VALIDATION 27 #define SNDRV_CTL_ELEM_ACCESS_SKIP_CHECK (1 << 2 7)27 #define SNDRV_CTL_ELEM_ACCESS_SKIP_CHECK (1 << 24) 28 28 #define snd_ctl_skip_validation(info) \ 29 29 ((info)->access & SNDRV_CTL_ELEM_ACCESS_SKIP_CHECK) … … 32 32 #define snd_ctl_skip_validation(info) true 33 33 #endif 34 35 /* kernel only - LED bits */ 36 #define SNDRV_CTL_ELEM_ACCESS_LED_SHIFT 25 37 #define SNDRV_CTL_ELEM_ACCESS_LED_MASK (7<<25) /* kernel three bits - LED group */ 38 #define SNDRV_CTL_ELEM_ACCESS_SPK_LED (1<<25) /* kernel speaker (output) LED flag */ 39 #define SNDRV_CTL_ELEM_ACCESS_MIC_LED (2<<25) /* kernel microphone (input) LED flag */ 34 40 35 41 enum { … … 109 115 }; 110 116 117 struct snd_ctl_layer_ops { 118 struct snd_ctl_layer_ops *next; 119 const char *module_name; 120 void (*lregister)(struct snd_card *card); 121 void (*ldisconnect)(struct snd_card *card); 122 void (*lnotify)(struct snd_card *card, unsigned int mask, struct snd_kcontrol *kctl, unsigned int ioff); 123 }; 124 111 125 #define snd_ctl_file(n) list_entry(n, struct snd_ctl_file, list) 112 126 … … 116 130 117 131 void snd_ctl_notify(struct snd_card * card, unsigned int mask, struct snd_ctl_elem_id * id); 132 void snd_ctl_notify_one(struct snd_card * card, unsigned int mask, struct snd_kcontrol * kctl, unsigned int ioff); 118 133 119 134 struct snd_kcontrol *snd_ctl_new1(const struct snd_kcontrol_new * kcontrolnew, void * private_data); … … 124 139 int snd_ctl_remove_id(struct snd_card * card, struct snd_ctl_elem_id *id); 125 140 int snd_ctl_rename_id(struct snd_card * card, struct snd_ctl_elem_id *src_id, struct snd_ctl_elem_id *dst_id); 126 int snd_ctl_activate_id(struct snd_card *card, struct snd_ctl_elem_id *id, 127 int active); 141 int snd_ctl_activate_id(struct snd_card *card, struct snd_ctl_elem_id *id, int active); 128 142 struct snd_kcontrol *snd_ctl_find_numid(struct snd_card * card, unsigned int numid); 129 143 struct snd_kcontrol *snd_ctl_find_id(struct snd_card * card, struct snd_ctl_elem_id *id); … … 140 154 #define snd_ctl_unregister_ioctl_compat(fcn) 141 155 #endif 156 157 int snd_ctl_request_layer(const char *module_name); 158 void snd_ctl_register_layer(struct snd_ctl_layer_ops *lops); 159 void snd_ctl_disconnect_layer(struct snd_ctl_layer_ops *lops); 142 160 143 161 int snd_ctl_get_preferred_subdevice(struct snd_card *card, int type); … … 255 273 256 274 /* 275 * Control LED trigger layer 276 */ 277 #define SND_CTL_LAYER_MODULE_LED "snd-ctl-led" 278 279 #if IS_MODULE(CONFIG_SND_CTL_LED) 280 static inline int snd_ctl_led_request(void) { return snd_ctl_request_layer(SND_CTL_LAYER_MODULE_LED); } 281 #else 282 static inline int snd_ctl_led_request(void) { return 0; } 283 #endif 284 285 /* 257 286 * Helper functions for jack-detection controls 258 287 */ -
TabularUnified GPL/branches/uniaud32-next/alsa-kernel/include/sound/core.h ¶
r693 r694 104 104 rwlock_t ctl_files_rwlock; /* ctl_files list lock */ 105 105 int controls_count; /* count of all controls */ 106 int user_ctl_count; /* count of all user controls */106 size_t user_ctl_alloc_size; // current memory allocation by user controls. 107 107 struct list_head controls; /* all controls for this card */ 108 108 struct list_head ctl_files; /* active control files */ -
TabularUnified GPL/branches/uniaud32-next/alsa-kernel/include/sound/hda_register.h ¶
r629 r694 141 141 #define AZX_MAX_BDL_ENTRIES (BDL_SIZE / 16) 142 142 #define AZX_MAX_FRAG 32 143 /* max buffer size - no h/w limit, you can increase as you like */ 144 #define AZX_MAX_BUF_SIZE (1024*1024*1024) 143 /* 144 * max buffer size - artificial 4MB limit per stream to avoid big allocations 145 * In theory it can be really big, but as it is per stream on systems with many streams memory could 146 * be quickly saturated if userspace requests maximum buffer size for each of them. 147 */ 148 #define AZX_MAX_BUF_SIZE (4*1024*1024) 145 149 146 150 /* RIRB int mask: overrun[2], response[0] */ -
TabularUnified GPL/branches/uniaud32-next/alsa-kernel/include/sound/simple_card_utils.h ¶
r692 r694 39 39 }; 40 40 41 struct prop_nums { 42 int cpus; 43 int codecs; 44 int platforms; 45 }; 46 41 47 struct asoc_simple_priv { 42 48 struct snd_soc_card snd_card; … … 44 50 struct asoc_simple_dai *cpu_dai; 45 51 struct asoc_simple_dai *codec_dai; 46 struct snd_soc_dai_link_component cpus; /* single cpu */47 struct snd_soc_dai_link_component codecs; /* single codec */48 struct snd_soc_dai_link_component platforms;52 struct snd_soc_dai_link_component *cpus; 53 struct snd_soc_dai_link_component *codecs; 54 struct snd_soc_dai_link_component *platforms; 49 55 struct asoc_simple_data adata; 50 56 struct snd_soc_codec_conf *codec_conf; 57 struct prop_nums num; 51 58 unsigned int mclk_fs; 52 59 } *dai_props; … … 55 62 struct snd_soc_dai_link *dai_link; 56 63 struct asoc_simple_dai *dais; 64 struct snd_soc_dai_link_component *dlcs; 65 struct snd_soc_dai_link_component dummy; 57 66 struct snd_soc_codec_conf *codec_conf; 58 67 struct gpio_desc *pa_gpio; … … 66 75 #define simple_priv_to_link(priv, i) (simple_priv_to_card(priv)->dai_link + (i)) 67 76 77 #define simple_props_to_dlc_cpu(props, i) ((props)->cpus + i) 78 #define simple_props_to_dlc_codec(props, i) ((props)->codecs + i) 79 #define simple_props_to_dlc_platform(props, i) ((props)->platforms + i) 80 81 #define simple_props_to_dai_cpu(props, i) ((props)->cpu_dai + i) 82 #define simple_props_to_dai_codec(props, i) ((props)->codec_dai + i) 83 #define simple_props_to_codec_conf(props, i) ((props)->codec_conf + i) 84 85 #define for_each_prop_dlc_cpus(props, i, cpu) \ 86 for ((i) = 0; \ 87 ((i) < (props)->num.cpus) && \ 88 ((cpu) = simple_props_to_dlc_cpu(props, i)); \ 89 (i)++) 90 #define for_each_prop_dlc_codecs(props, i, codec) \ 91 for ((i) = 0; \ 92 ((i) < (props)->num.codecs) && \ 93 ((codec) = simple_props_to_dlc_codec(props, i)); \ 94 (i)++) 95 #define for_each_prop_dlc_platforms(props, i, platform) \ 96 for ((i) = 0; \ 97 ((i) < (props)->num.platforms) && \ 98 ((platform) = simple_props_to_dlc_platform(props, i)); \ 99 (i)++) 100 #define for_each_prop_codec_conf(props, i, conf) \ 101 for ((i) = 0; \ 102 ((i) < (props)->num.codecs) && \ 103 (props)->codec_conf && \ 104 ((conf) = simple_props_to_codec_conf(props, i)); \ 105 (i)++) 106 107 #define for_each_prop_dai_cpu(props, i, cpu) \ 108 for ((i) = 0; \ 109 ((i) < (props)->num.cpus) && \ 110 ((cpu) = simple_props_to_dai_cpu(props, i)); \ 111 (i)++) 112 #define for_each_prop_dai_codec(props, i, codec) \ 113 for ((i) = 0; \ 114 ((i) < (props)->num.codecs) && \ 115 ((codec) = simple_props_to_dai_codec(props, i)); \ 116 (i)++) 117 118 #define SNDRV_MAX_LINKS 128 119 68 120 struct link_info { 69 int dais; /* number of dai */70 121 int link; /* number of link */ 71 int conf; /* number of codec_conf */72 122 int cpu; /* turn for CPU / Codec */ 123 struct prop_nums num[SNDRV_MAX_LINKS]; 73 124 }; 74 125 … … 85 136 char *prefix); 86 137 87 #define asoc_simple_parse_clk_cpu(dev, node, dai_link, simple_dai) \88 asoc_simple_parse_clk(dev, node, simple_dai, dai_link->cpus)89 #define asoc_simple_parse_clk_codec(dev, node, dai_link, simple_dai) \90 asoc_simple_parse_clk(dev, node, simple_dai, dai_link->codecs)91 138 int asoc_simple_parse_clk(struct device *dev, 92 139 struct device_node *node, … … 101 148 struct snd_pcm_hw_params *params); 102 149 103 #define asoc_simple_parse_cpu(node, dai_link, is_single_link) \104 asoc_simple_parse_dai(node, dai_link->cpus, is_single_link)105 #define asoc_simple_parse_codec(node, dai_link) \106 asoc_simple_parse_dai(node, dai_link->codecs, NULL)107 #define asoc_simple_parse_platform(node, dai_link) \108 asoc_simple_parse_dai(node, dai_link->platforms, NULL)109 110 150 #define asoc_simple_parse_tdm(np, dai) \ 111 151 snd_soc_of_parse_tdm_slot(np, &(dai)->tx_slot_mask, \ … … 114 154 &(dai)->slot_width); 115 155 116 void asoc_simple_canonicalize_platform(struct snd_soc_dai_link *dai_link); 117 void asoc_simple_canonicalize_cpu(struct snd_soc_dai_link *dai_link, 118 int is_single_links); 156 void asoc_simple_canonicalize_platform(struct snd_soc_dai_link_component *platforms, 157 struct snd_soc_dai_link_component *cpus); 158 void asoc_simple_canonicalize_cpu(struct snd_soc_dai_link_component *cpus, 159 int is_single_links); 119 160 120 161 int asoc_simple_clean_reference(struct snd_soc_card *card); … … 122 163 void asoc_simple_convert_fixup(struct asoc_simple_data *data, 123 164 struct snd_pcm_hw_params *params); 124 void asoc_simple_parse_convert(struct device *dev, 125 struct device_node *np, char *prefix, 165 void asoc_simple_parse_convert(struct device_node *np, char *prefix, 126 166 struct asoc_simple_data *data); 127 167 … … 138 178 int asoc_simple_init_priv(struct asoc_simple_priv *priv, 139 179 struct link_info *li); 180 int asoc_simple_remove(struct platform_device *pdev); 181 182 int asoc_graph_card_probe(struct snd_soc_card *card); 140 183 141 184 #ifdef DEBUG … … 153 196 dev_dbg(dev, "%s dai name = %s\n", 154 197 name, dai->name); 155 if (dai->sysclk)156 dev_dbg(dev, "%s sysclk = %d\n",157 name, dai->sysclk);158 159 dev_dbg(dev, "%s direction = %s\n",160 name, dai->clk_direction ? "OUT" : "IN");161 198 162 199 if (dai->slots) … … 170 207 if (dai->clk) 171 208 dev_dbg(dev, "%s clk %luHz\n", name, clk_get_rate(dai->clk)); 209 if (dai->sysclk) 210 dev_dbg(dev, "%s sysclk = %dHz\n", 211 name, dai->sysclk); 212 if (dai->clk || dai->sysclk) 213 dev_dbg(dev, "%s direction = %s\n", 214 name, dai->clk_direction ? "OUT" : "IN"); 172 215 } 173 216 … … 185 228 struct simple_dai_props *props = simple_priv_to_props(priv, i); 186 229 struct snd_soc_dai_link *link = simple_priv_to_link(priv, i); 230 struct asoc_simple_dai *dai; 231 struct snd_soc_codec_conf *cnf; 232 int j; 187 233 188 234 dev_dbg(dev, "DAI%d\n", i); 189 235 190 asoc_simple_debug_dai(priv, "cpu", props->cpu_dai); 191 asoc_simple_debug_dai(priv, "codec", props->codec_dai); 236 dev_dbg(dev, "cpu num = %d\n", link->num_cpus); 237 for_each_prop_dai_cpu(props, j, dai) 238 asoc_simple_debug_dai(priv, "cpu", dai); 239 dev_dbg(dev, "codec num = %d\n", link->num_codecs); 240 for_each_prop_dai_codec(props, j, dai) 241 asoc_simple_debug_dai(priv, "codec", dai); 192 242 193 243 if (link->name) 194 244 dev_dbg(dev, "dai name = %s\n", link->name); 195 196 dev_dbg(dev, "dai format = %04x\n", link->dai_fmt); 197 245 if (link->dai_fmt) 246 dev_dbg(dev, "dai format = %04x\n", link->dai_fmt); 198 247 if (props->adata.convert_rate) 199 dev_dbg(dev, "convert_rate = %d\n", 200 props->adata.convert_rate); 248 dev_dbg(dev, "convert_rate = %d\n", props->adata.convert_rate); 201 249 if (props->adata.convert_channels) 202 dev_dbg(dev, "convert_channels = %d\n", 203 props->adata.convert_channels); 204 if (props->codec_conf && props->codec_conf->name_prefix) 205 dev_dbg(dev, "name prefix = %s\n", 206 props->codec_conf->name_prefix); 250 dev_dbg(dev, "convert_channels = %d\n", props->adata.convert_channels); 251 for_each_prop_codec_conf(props, j, cnf) 252 if (cnf->name_prefix) 253 dev_dbg(dev, "name prefix = %s\n", cnf->name_prefix); 207 254 if (props->mclk_fs) 208 dev_dbg(dev, "mclk-fs = %d\n", 209 props->mclk_fs); 255 dev_dbg(dev, "mclk-fs = %d\n", props->mclk_fs); 210 256 } 211 257 } -
TabularUnified GPL/branches/uniaud32-next/alsa-kernel/include/sound/soc-acpi.h ¶
r693 r694 64 64 * @link_mask: links enabled on the board 65 65 * @links: array of link _ADR descriptors, null terminated 66 * @num_dai_drivers: number of elements in @dai_drivers 67 * @dai_drivers: pointer to dai_drivers, used e.g. in nocodec mode 66 68 */ 67 69 struct snd_soc_acpi_mach_params { … … 73 75 u32 link_mask; 74 76 const struct snd_soc_acpi_link_adr *links; 77 u32 num_dai_drivers; 78 struct snd_soc_dai_driver *dai_drivers; 75 79 }; 76 80 -
TabularUnified GPL/branches/uniaud32-next/alsa-kernel/include/sound/soc-component.h ¶
r693 r694 102 102 /* DT */ 103 103 int (*of_xlate_dai_name)(struct snd_soc_component *component, 104 struct of_phandle_args *args,104 const struct of_phandle_args *args, 105 105 const char **dai_name); 106 106 int (*of_xlate_dai_id)(struct snd_soc_component *comment, … … 147 147 struct snd_pcm_substream *substream, 148 148 struct vm_area_struct *vma); 149 int (*ack)(struct snd_soc_component *component, 150 struct snd_pcm_substream *substream); 149 151 150 152 const struct snd_compress_ops *compress_ops; … … 337 339 struct snd_soc_aux_dev *aux); 338 340 int snd_soc_component_init(struct snd_soc_component *component); 341 int snd_soc_component_is_dummy(struct snd_soc_component *component); 339 342 340 343 /* component IO */ … … 451 454 struct device_node *ep); 452 455 int snd_soc_component_of_xlate_dai_name(struct snd_soc_component *component, 453 struct of_phandle_args *args,456 const struct of_phandle_args *args, 454 457 const char **dai_name); 455 458 int snd_soc_component_compr_open(struct snd_compr_stream *cstream); … … 499 502 void snd_soc_pcm_component_pm_runtime_put(struct snd_soc_pcm_runtime *rtd, 500 503 void *stream, int rollback); 504 int snd_soc_pcm_component_ack(struct snd_pcm_substream *substream); 501 505 502 506 #endif /* __SOC_COMPONENT_H */ -
TabularUnified GPL/branches/uniaud32-next/alsa-kernel/include/sound/soc-dai.h ¶
r693 r694 82 82 #define SND_SOC_DAIFMT_CBC_CFP (2 << 12) /* codec clk consumer & frame provider */ 83 83 #define SND_SOC_DAIFMT_CBP_CFC (3 << 12) /* codec clk provider & frame consumer */ 84 #define SND_SOC_DAIFMT_CBC_CFC (4 << 12) /* codec clk consumer & frame follower */84 #define SND_SOC_DAIFMT_CBC_CFC (4 << 12) /* codec clk consumer & frame consumer */ 85 85 86 86 /* previous definitions kept for backwards-compatibility, do not use in new contributions */ -
TabularUnified GPL/branches/uniaud32-next/alsa-kernel/include/sound/soc-dpcm.h ¶
r629 r694 150 150 int stream, struct snd_soc_dapm_widget_list **list, int new); 151 151 int dpcm_be_dai_startup(struct snd_soc_pcm_runtime *fe, int stream); 152 int dpcm_be_dai_shutdown(struct snd_soc_pcm_runtime *fe, int stream); 152 void dpcm_be_dai_stop(struct snd_soc_pcm_runtime *fe, int stream, 153 int do_hw_free, struct snd_soc_dpcm *last); 153 154 void dpcm_be_disconnect(struct snd_soc_pcm_runtime *fe, int stream); 154 155 void dpcm_clear_pending_state(struct snd_soc_pcm_runtime *fe, int stream); 155 intdpcm_be_dai_hw_free(struct snd_soc_pcm_runtime *fe, int stream);156 void dpcm_be_dai_hw_free(struct snd_soc_pcm_runtime *fe, int stream); 156 157 int dpcm_be_dai_hw_params(struct snd_soc_pcm_runtime *fe, int tream); 157 158 int dpcm_be_dai_trigger(struct snd_soc_pcm_runtime *fe, int stream, int cmd); … … 160 161 int event); 161 162 163 #define dpcm_be_dai_startup_rollback(fe, stream, last) \ 164 dpcm_be_dai_stop(fe, stream, 0, last) 165 #define dpcm_be_dai_startup_unwind(fe, stream) dpcm_be_dai_stop(fe, stream, 0, NULL) 166 #define dpcm_be_dai_shutdown(fe, stream) dpcm_be_dai_stop(fe, stream, 1, NULL) 167 162 168 #endif -
TabularUnified GPL/branches/uniaud32-next/alsa-kernel/include/sound/soc.h ¶
r693 r694 713 713 unsigned int ignore:1; 714 714 715 /* This flag will reorder stop sequence. By enabling this flag 716 * DMA controller stop sequence will be invoked first followed by 717 * CPU DAI driver stop sequence 718 */ 719 unsigned int stop_dma_first:1; 720 715 721 #ifdef CONFIG_SND_SOC_TOPOLOGY 716 722 struct snd_soc_dobj dobj; /* For topology */ 717 723 #endif 718 724 }; 725 726 static inline struct snd_soc_dai_link_component* 727 asoc_link_to_cpu(struct snd_soc_dai_link *link, int n) { 728 return &(link)->cpus[n]; 729 } 730 731 static inline struct snd_soc_dai_link_component* 732 asoc_link_to_codec(struct snd_soc_dai_link *link, int n) { 733 return &(link)->codecs[n]; 734 } 735 736 static inline struct snd_soc_dai_link_component* 737 asoc_link_to_platform(struct snd_soc_dai_link *link, int n) { 738 return &(link)->platforms[n]; 739 } 740 719 741 #define for_each_link_codecs(link, i, codec) \ 720 742 for ((i) = 0; \ 721 ((i) < link->num_codecs) && ((codec) = &link->codecs[i]); \ 743 ((i) < link->num_codecs) && \ 744 ((codec) = asoc_link_to_codec(link, i)); \ 722 745 (i)++) 723 746 … … 725 748 for ((i) = 0; \ 726 749 ((i) < link->num_platforms) && \ 727 ((platform) = &link->platforms[i]);\750 ((platform) = asoc_link_to_platform(link, i)); \ 728 751 (i)++) 729 752 730 753 #define for_each_link_cpus(link, i, cpu) \ 731 754 for ((i) = 0; \ 732 ((i) < link->num_cpus) && ((cpu) = &link->cpus[i]); \ 755 ((i) < link->num_cpus) && \ 756 ((cpu) = asoc_link_to_cpu(link, i)); \ 733 757 (i)++) 734 758 … … 1220 1244 struct device_node **framemaster); 1221 1245 int snd_soc_get_dai_id(struct device_node *ep); 1222 int snd_soc_get_dai_name( struct of_phandle_args *args,1246 int snd_soc_get_dai_name(const struct of_phandle_args *args, 1223 1247 const char **dai_name); 1224 1248 int snd_soc_of_get_dai_name(struct device_node *of_node, … … 1263 1287 /* set platform name for each dailink */ 1264 1288 for_each_card_prelinks(card, i, dai_link) { 1289 /* only single platform is supported for now */ 1290 if (dai_link->num_platforms != 1) 1291 return -EINVAL; 1292 1293 if (!dai_link->platforms) 1294 return -EINVAL; 1295 1265 1296 name = devm_kstrdup(card->dev, platform_name, GFP_KERNEL); 1266 1297 if (!name) 1267 1298 return -ENOMEM; 1268 1299 1269 if (!dai_link->platforms)1270 return -EINVAL;1271 1272 1300 /* only single platform is supported for now */ 1273 1301 dai_link->platforms->name = name; -
TabularUnified GPL/branches/uniaud32-next/alsa-kernel/include/sound/sof.h ¶
r692 r694 101 101 }; 102 102 103 int sof_nocodec_setup(struct device *dev, const struct snd_sof_dsp_ops *ops, 104 int (*pcm_dai_link_fixup)(struct snd_soc_pcm_runtime *rtd, 105 struct snd_pcm_hw_params *params)); 103 int sof_dai_get_mclk(struct snd_soc_pcm_runtime *rtd); 106 104 107 105 #endif -
TabularUnified GPL/branches/uniaud32-next/alsa-kernel/include/sound/version.h ¶
r693 r694 1 1 /* include/version.h */ 2 #define CONFIG_SND_VERSION "5.1 2.19"2 #define CONFIG_SND_VERSION "5.13.10" 3 3 #define CONFIG_SND_DATE "" -
TabularUnified GPL/branches/uniaud32-next/alsa-kernel/isa/sb/sb16_csp.c ¶
r693 r694 816 816 snd_sbmixer_write(p->chip, SB_DSP4_PCM_DEV, mixL & 0x7); 817 817 snd_sbmixer_write(p->chip, SB_DSP4_PCM_DEV + 1, mixR & 0x7); 818 spin_unlock_irqrestore(&p->chip->mixer_lock, flags); 818 819 819 820 spin_lock(&p->chip->reg_lock); … … 855 856 856 857 /* restore PCM volume */ 858 spin_lock_irqsave(&p->chip->mixer_lock, flags); 857 859 snd_sbmixer_write(p->chip, SB_DSP4_PCM_DEV, mixL); 858 860 snd_sbmixer_write(p->chip, SB_DSP4_PCM_DEV + 1, mixR); … … 880 882 snd_sbmixer_write(p->chip, SB_DSP4_PCM_DEV, mixL & 0x7); 881 883 snd_sbmixer_write(p->chip, SB_DSP4_PCM_DEV + 1, mixR & 0x7); 884 spin_unlock_irqrestore(&p->chip->mixer_lock, flags); 882 885 883 886 spin_lock(&p->chip->reg_lock); … … 894 897 895 898 /* restore PCM volume */ 899 spin_lock_irqsave(&p->chip->mixer_lock, flags); 896 900 snd_sbmixer_write(p->chip, SB_DSP4_PCM_DEV, mixL); 897 901 snd_sbmixer_write(p->chip, SB_DSP4_PCM_DEV + 1, mixR); -
TabularUnified GPL/branches/uniaud32-next/alsa-kernel/isa/sb/sb8.c ¶
r693 r694 94 94 card->private_free = snd_sb8_free; 95 95 96 /* block the 0x388 port to avoid PnP conflicts */ 96 /* 97 * Block the 0x388 port to avoid PnP conflicts. 98 * No need to check this value after request_region, 99 * as we never do anything with it. 100 */ 97 101 acard->fm_res = request_region(0x388, 4, "SoundBlaster FM"); 98 102 -
TabularUnified GPL/branches/uniaud32-next/alsa-kernel/pci/hda/hda_auto_parser.h ¶
r615 r694 28 28 29 29 #define AUTO_CFG_MAX_OUTS HDA_MAX_OUTS 30 #define AUTO_CFG_MAX_INS 830 #define AUTO_CFG_MAX_INS 18 31 31 32 32 struct auto_pin_cfg_item { -
TabularUnified GPL/branches/uniaud32-next/alsa-kernel/pci/hda/hda_codec.c ¶
r693 r694 1950 1950 * @suffix: suffix string to each follower name (optional) 1951 1951 * @init_follower_vol: initialize followers to unmute/0dB 1952 * @access: kcontrol access rights 1952 1953 * @ctl_ret: store the vmaster kcontrol in return 1953 1954 * … … 1964 1965 unsigned int *tlv, const char * const *followers, 1965 1966 const char *suffix, bool init_follower_vol, 1966 struct snd_kcontrol **ctl_ret)1967 unsigned int access, struct snd_kcontrol **ctl_ret) 1967 1968 { 1968 1969 struct snd_kcontrol *kctl; … … 1980 1981 if (!kctl) 1981 1982 return -ENOMEM; 1983 kctl->vd[0].access |= access; 1982 1984 err = snd_hda_ctl_add(codec, 0, kctl); 1983 1985 if (err < 0) … … 2011 2013 EXPORT_SYMBOL_GPL(__snd_hda_add_vmaster); 2012 2014 2013 /*2014 * mute-LED control using vmaster2015 */2016 static int vmaster_mute_mode_info(struct snd_kcontrol *kcontrol,2017 struct snd_ctl_elem_info *uinfo)2018 {2019 static const char * const texts[] = {2020 "On", "Off", "Follow Master"2021 };2022 2023 return snd_ctl_enum_info(uinfo, 1, 3, texts);2024 }2025 2026 static int vmaster_mute_mode_get(struct snd_kcontrol *kcontrol,2027 struct snd_ctl_elem_value *ucontrol)2028 {2029 struct hda_vmaster_mute_hook *hook = snd_kcontrol_chip(kcontrol);2030 ucontrol->value.enumerated.item[0] = hook->mute_mode;2031 return 0;2032 }2033 2034 static int vmaster_mute_mode_put(struct snd_kcontrol *kcontrol,2035 struct snd_ctl_elem_value *ucontrol)2036 {2037 struct hda_vmaster_mute_hook *hook = snd_kcontrol_chip(kcontrol);2038 unsigned int old_mode = hook->mute_mode;2039 2040 hook->mute_mode = ucontrol->value.enumerated.item[0];2041 if (hook->mute_mode > HDA_VMUTE_FOLLOW_MASTER)2042 hook->mute_mode = HDA_VMUTE_FOLLOW_MASTER;2043 if (old_mode == hook->mute_mode)2044 return 0;2045 snd_hda_sync_vmaster_hook(hook);2046 return 1;2047 }2048 2049 static const struct snd_kcontrol_new vmaster_mute_mode = {2050 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,2051 .name = "Mute-LED Mode",2052 .info = vmaster_mute_mode_info,2053 .get = vmaster_mute_mode_get,2054 .put = vmaster_mute_mode_put,2055 };2056 2057 2015 /* meta hook to call each driver's vmaster hook */ 2058 2016 static void vmaster_hook(void *private_data, int enabled) … … 2060 2018 struct hda_vmaster_mute_hook *hook = private_data; 2061 2019 2062 if (hook->mute_mode != HDA_VMUTE_FOLLOW_MASTER)2063 enabled = hook->mute_mode;2064 2020 hook->hook(hook->codec, enabled); 2065 2021 } 2066 2022 2067 2023 /** 2068 * snd_hda_add_vmaster_hook - Add a vmaster h ook for mute-LED2024 * snd_hda_add_vmaster_hook - Add a vmaster hw specific hook 2069 2025 * @codec: the HDA codec 2070 2026 * @hook: the vmaster hook object 2071 * @expose_enum_ctl: flag to create an enum ctl 2072 * 2073 * Add a mute-LED hook with the given vmaster switch kctl. 2074 * When @expose_enum_ctl is set, "Mute-LED Mode" control is automatically 2075 * created and associated with the given hook. 2027 * 2028 * Add a hw specific hook (like EAPD) with the given vmaster switch kctl. 2076 2029 */ 2077 2030 int snd_hda_add_vmaster_hook(struct hda_codec *codec, 2078 struct hda_vmaster_mute_hook *hook, 2079 bool expose_enum_ctl) 2080 { 2081 struct snd_kcontrol *kctl; 2082 2031 struct hda_vmaster_mute_hook *hook) 2032 { 2083 2033 if (!hook->hook || !hook->sw_kctl) 2084 2034 return 0; 2085 2035 hook->codec = codec; 2086 hook->mute_mode = HDA_VMUTE_FOLLOW_MASTER;2087 2036 snd_ctl_add_vmaster_hook(hook->sw_kctl, vmaster_hook, hook); 2088 if (!expose_enum_ctl) 2089 return 0; 2090 kctl = snd_ctl_new1(&vmaster_mute_mode, hook); 2091 if (!kctl) 2092 return -ENOMEM; 2093 return snd_hda_ctl_add(codec, 0, kctl); 2037 return 0; 2094 2038 } 2095 2039 EXPORT_SYMBOL_GPL(snd_hda_add_vmaster_hook); -
TabularUnified GPL/branches/uniaud32-next/alsa-kernel/pci/hda/hda_generic.c ¶
r693 r694 985 985 if (get_amp_nid_(val)) 986 986 knew->subdevice = HDA_SUBDEV_AMP_FLAG; 987 if (knew->access == 0) 988 knew->access = SNDRV_CTL_ELEM_ACCESS_READWRITE; 987 989 knew->private_value = val; 988 990 return knew; … … 3530 3532 .iface = SNDRV_CTL_ELEM_IFACE_MIXER, 3531 3533 .name = "Capture Switch", 3534 .access = SNDRV_CTL_ELEM_ACCESS_READWRITE, 3532 3535 .info = cap_sw_info, 3533 3536 .get = cap_sw_get, … … 3636 3639 if (!knew) 3637 3640 return -ENOMEM; 3638 if (is_switch) 3641 if (is_switch) { 3639 3642 knew->put = cap_single_sw_put; 3643 if (spec->mic_mute_led) 3644 knew->access |= SNDRV_CTL_ELEM_ACCESS_MIC_LED; 3645 } 3640 3646 if (!inv_dmic) 3641 3647 return 0; … … 3652 3658 if (!knew) 3653 3659 return -ENOMEM; 3654 if (is_switch) 3660 if (is_switch) { 3655 3661 knew->put = cap_single_sw_put; 3662 if (spec->mic_mute_led) 3663 knew->access |= SNDRV_CTL_ELEM_ACCESS_MIC_LED; 3664 } 3656 3665 return 0; 3657 3666 } … … 3694 3703 knew->private_value = sw_ctl; 3695 3704 knew->subdevice = HDA_SUBDEV_AMP_FLAG; 3705 if (spec->mic_mute_led) 3706 knew->access |= SNDRV_CTL_ELEM_ACCESS_MIC_LED; 3696 3707 } 3697 3708 return 0; … … 3935 3946 } 3936 3947 3937 static void vmaster_update_mute_led(void *private_data, int enabled)3938 {3939 ledtrig_audio_set(LED_AUDIO_MUTE, enabled ? LED_OFF : LED_ON);3940 }3941 3942 3948 /** 3943 3949 * snd_hda_gen_add_mute_led_cdev - Create a LED classdev and enable as vmaster mute LED … … 3963 3969 codec_err(codec, "vmaster hook already present before cdev!\n"); 3964 3970 3965 spec->vmaster_mute.hook = vmaster_update_mute_led; 3966 spec->vmaster_mute_enum = 1; 3971 spec->vmaster_mute_led = 1; 3967 3972 return 0; 3968 3973 } 3969 3974 EXPORT_SYMBOL_GPL(snd_hda_gen_add_mute_led_cdev); 3970 3971 /*3972 * mic mute LED hook helpers3973 */3974 enum {3975 MICMUTE_LED_ON,3976 MICMUTE_LED_OFF,3977 MICMUTE_LED_FOLLOW_CAPTURE,3978 MICMUTE_LED_FOLLOW_MUTE,3979 };3980 3981 static void call_micmute_led_update(struct hda_codec *codec)3982 {3983 struct hda_gen_spec *spec = codec->spec;3984 unsigned int val;3985 3986 switch (spec->micmute_led.led_mode) {3987 case MICMUTE_LED_ON:3988 val = 1;3989 break;3990 case MICMUTE_LED_OFF:3991 val = 0;3992 break;3993 case MICMUTE_LED_FOLLOW_CAPTURE:3994 val = !!spec->micmute_led.capture;3995 break;3996 case MICMUTE_LED_FOLLOW_MUTE:3997 default:3998 val = !spec->micmute_led.capture;3999 break;4000 }4001 4002 if (val == spec->micmute_led.led_value)4003 return;4004 spec->micmute_led.led_value = val;4005 ledtrig_audio_set(LED_AUDIO_MICMUTE,4006 spec->micmute_led.led_value ? LED_ON : LED_OFF);4007 }4008 4009 static void update_micmute_led(struct hda_codec *codec,4010 struct snd_kcontrol *kcontrol,4011 struct snd_ctl_elem_value *ucontrol)4012 {4013 struct hda_gen_spec *spec = codec->spec;4014 unsigned int mask;4015 4016 if (spec->micmute_led.old_hook)4017 spec->micmute_led.old_hook(codec, kcontrol, ucontrol);4018 4019 if (!ucontrol)4020 return;4021 mask = 1U << snd_ctl_get_ioffidx(kcontrol, &ucontrol->id);4022 if (!strcmp("Capture Switch", ucontrol->id.name)) {4023 /* TODO: How do I verify if it's a mono or stereo here? */4024 if (ucontrol->value.integer.value[0] ||4025 ucontrol->value.integer.value[1])4026 spec->micmute_led.capture |= mask;4027 else4028 spec->micmute_led.capture &= ~mask;4029 call_micmute_led_update(codec);4030 }4031 }4032 4033 static int micmute_led_mode_info(struct snd_kcontrol *kcontrol,4034 struct snd_ctl_elem_info *uinfo)4035 {4036 static const char * const texts[] = {4037 "On", "Off", "Follow Capture", "Follow Mute",4038 };4039 4040 return snd_ctl_enum_info(uinfo, 1, ARRAY_SIZE(texts), texts);4041 }4042 4043 static int micmute_led_mode_get(struct snd_kcontrol *kcontrol,4044 struct snd_ctl_elem_value *ucontrol)4045 {4046 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);4047 struct hda_gen_spec *spec = codec->spec;4048 4049 ucontrol->value.enumerated.item[0] = spec->micmute_led.led_mode;4050 return 0;4051 }4052 4053 static int micmute_led_mode_put(struct snd_kcontrol *kcontrol,4054 struct snd_ctl_elem_value *ucontrol)4055 {4056 struct hda_codec *codec = snd_kcontrol_chip(kcontrol);4057 struct hda_gen_spec *spec = codec->spec;4058 unsigned int mode;4059 4060 mode = ucontrol->value.enumerated.item[0];4061 if (mode > MICMUTE_LED_FOLLOW_MUTE)4062 mode = MICMUTE_LED_FOLLOW_MUTE;4063 if (mode == spec->micmute_led.led_mode)4064 return 0;4065 spec->micmute_led.led_mode = mode;4066 call_micmute_led_update(codec);4067 return 1;4068 }4069 4070 static const struct snd_kcontrol_new micmute_led_mode_ctl = {4071 .iface = SNDRV_CTL_ELEM_IFACE_MIXER,4072 .name = "Mic Mute-LED Mode",4073 .info = micmute_led_mode_info,4074 .get = micmute_led_mode_get,4075 .put = micmute_led_mode_put,4076 };4077 4078 /* Set up the capture sync hook for controlling the mic-mute LED */4079 static int add_micmute_led_hook(struct hda_codec *codec)4080 {4081 struct hda_gen_spec *spec = codec->spec;4082 4083 spec->micmute_led.led_mode = MICMUTE_LED_FOLLOW_MUTE;4084 spec->micmute_led.capture = 0;4085 spec->micmute_led.led_value = -1;4086 spec->micmute_led.old_hook = spec->cap_sync_hook;4087 spec->cap_sync_hook = update_micmute_led;4088 if (!snd_hda_gen_add_kctl(spec, NULL, &micmute_led_mode_ctl))4089 return -ENOMEM;4090 return 0;4091 }4092 3975 4093 3976 /** … … 4109 3992 enum led_brightness)) 4110 3993 { 3994 struct hda_gen_spec *spec = codec->spec; 4111 3995 int err; 4112 3996 … … 4119 4003 } 4120 4004 4121 return add_micmute_led_hook(codec); 4005 spec->mic_mute_led = 1; 4006 return 0; 4122 4007 } 4123 4008 EXPORT_SYMBOL_GPL(snd_hda_gen_add_micmute_led_cdev); … … 5078 4963 parse_user_hints(codec); 5079 4964 4965 if (spec->vmaster_mute_led || spec->mic_mute_led) 4966 snd_ctl_led_request(); 4967 5080 4968 if (spec->mixer_nid && !spec->mixer_merge_nid) 5081 4969 spec->mixer_merge_nid = spec->mixer_nid; … … 5309 5197 err = snd_hda_add_vmaster(codec, "Master Playback Volume", 5310 5198 spec->vmaster_tlv, follower_pfxs, 5311 "Playback Volume" );5199 "Playback Volume", 0); 5312 5200 if (err < 0) 5313 5201 return err; … … 5317 5205 err = __snd_hda_add_vmaster(codec, "Master Playback Switch", 5318 5206 NULL, follower_pfxs, 5319 "Playback Switch", 5320 true, &spec->vmaster_mute.sw_kctl); 5207 "Playback Switch", true, 5208 spec->vmaster_mute_led ? 5209 SNDRV_CTL_ELEM_ACCESS_SPK_LED : 0, 5210 &spec->vmaster_mute.sw_kctl); 5321 5211 if (err < 0) 5322 5212 return err; 5323 5213 if (spec->vmaster_mute.hook) { 5324 snd_hda_add_vmaster_hook(codec, &spec->vmaster_mute, 5325 spec->vmaster_mute_enum); 5214 snd_hda_add_vmaster_hook(codec, &spec->vmaster_mute); 5326 5215 snd_hda_sync_vmaster_hook(&spec->vmaster_mute); 5327 5216 } -
TabularUnified GPL/branches/uniaud32-next/alsa-kernel/pci/hda/hda_generic.h ¶
r629 r694 84 84 extern const struct badness_table hda_main_out_badness; 85 85 extern const struct badness_table hda_extra_out_badness; 86 87 struct hda_micmute_hook {88 unsigned int led_mode;89 unsigned int capture;90 unsigned int led_value;91 void (*old_hook)(struct hda_codec *codec,92 struct snd_kcontrol *kcontrol,93 struct snd_ctl_elem_value *ucontrol);94 };95 86 96 87 struct hda_gen_spec { … … 230 221 unsigned int own_eapd_ctl:1; /* set EAPD by own function */ 231 222 unsigned int keep_eapd_on:1; /* don't turn off EAPD automatically */ 232 unsigned int vmaster_mute_enum:1; /* add vmaster mute mode enum */ 223 unsigned int vmaster_mute_led:1; /* add SPK-LED flag to vmaster mute switch */ 224 unsigned int mic_mute_led:1; /* add MIC-LED flag to capture mute switch */ 233 225 unsigned int indep_hp:1; /* independent HP supported */ 234 226 unsigned int prefer_hp_amp:1; /* enable HP amp for speaker if any */ … … 285 277 struct snd_kcontrol *kcontrol, 286 278 struct snd_ctl_elem_value *ucontrol); 287 288 /* mic mute LED hook; called via cap_sync_hook */289 struct hda_micmute_hook micmute_led;290 279 291 280 /* PCM hooks */ -
TabularUnified GPL/branches/uniaud32-next/alsa-kernel/pci/hda/hda_jack.c ¶
r693 r694 388 388 } 389 389 EXPORT_SYMBOL_GPL(snd_hda_jack_set_gating_jack); 390 391 /** 392 * snd_hda_jack_bind_keymap - bind keys generated from one NID to another jack. 393 * @codec: the HDA codec 394 * @key_nid: key event is generated by this pin NID 395 * @keymap: map of key type and key code 396 * @jack_nid: key reports to the jack of this pin NID 397 * 398 * This function is used in the case of key is generated from one NID while is 399 * reported to the jack of another NID. 400 */ 401 int snd_hda_jack_bind_keymap(struct hda_codec *codec, hda_nid_t key_nid, 402 const struct hda_jack_keymap *keymap, 403 hda_nid_t jack_nid) 404 { 405 const struct hda_jack_keymap *map; 406 struct hda_jack_tbl *key_gen = snd_hda_jack_tbl_get(codec, key_nid); 407 struct hda_jack_tbl *report_to = snd_hda_jack_tbl_get(codec, jack_nid); 408 409 WARN_ON(codec->dp_mst); 410 411 if (!key_gen || !report_to || !report_to->jack) 412 return -EINVAL; 413 414 key_gen->key_report_jack = jack_nid; 415 416 if (keymap) 417 for (map = keymap; map->type; map++) 418 snd_jack_set_key(report_to->jack, map->type, map->key); 419 420 return 0; 421 } 422 EXPORT_SYMBOL_GPL(snd_hda_jack_bind_keymap); 423 424 /** 425 * snd_hda_jack_set_button_state - report button event to the hda_jack_tbl button_state. 426 * @codec: the HDA codec 427 * @jack_nid: the button event reports to the jack_tbl of this NID 428 * @button_state: the button event captured by codec 429 * 430 * Codec driver calls this function to report the button event. 431 */ 432 void snd_hda_jack_set_button_state(struct hda_codec *codec, hda_nid_t jack_nid, 433 int button_state) 434 { 435 struct hda_jack_tbl *jack = snd_hda_jack_tbl_get(codec, jack_nid); 436 437 if (!jack) 438 return; 439 440 if (jack->key_report_jack) { 441 struct hda_jack_tbl *report_to = 442 snd_hda_jack_tbl_get(codec, jack->key_report_jack); 443 444 if (report_to) { 445 report_to->button_state = button_state; 446 return; 447 } 448 } 449 450 jack->button_state = button_state; 451 } 452 EXPORT_SYMBOL_GPL(snd_hda_jack_set_button_state); 390 453 391 454 /** … … 652 715 if (!event) 653 716 return; 654 event->jack_dirty = 1; 717 718 if (event->key_report_jack) { 719 struct hda_jack_tbl *report_to = 720 snd_hda_jack_tbl_get_mst(codec, event->key_report_jack, 721 event->dev_id); 722 if (report_to) 723 report_to->jack_dirty = 1; 724 } else 725 event->jack_dirty = 1; 655 726 656 727 call_jack_callback(codec, res, event); -
TabularUnified GPL/branches/uniaud32-next/alsa-kernel/pci/hda/hda_jack.h ¶
r629 r694 41 41 hda_nid_t gating_jack; /* valid when gating jack plugged */ 42 42 hda_nid_t gated_jack; /* gated is dependent on this jack */ 43 hda_nid_t key_report_jack; /* key reports to this jack */ 43 44 int type; 44 45 int button_state; … … 99 100 int snd_hda_jack_set_gating_jack(struct hda_codec *codec, hda_nid_t gated_nid, 100 101 hda_nid_t gating_nid); 102 103 int snd_hda_jack_bind_keymap(struct hda_codec *codec, hda_nid_t key_nid, 104 const struct hda_jack_keymap *keymap, 105 hda_nid_t jack_nid); 106 107 void snd_hda_jack_set_button_state(struct hda_codec *codec, hda_nid_t jack_nid, 108 int button_state); 101 109 102 110 u32 snd_hda_jack_pin_sense(struct hda_codec *codec, hda_nid_t nid, int dev_id); -
TabularUnified GPL/branches/uniaud32-next/alsa-kernel/pci/hda/hda_local.h ¶
r629 r694 132 132 unsigned int *tlv, const char * const *followers, 133 133 const char *suffix, bool init_follower_vol, 134 struct snd_kcontrol **ctl_ret);135 #define snd_hda_add_vmaster(codec, name, tlv, followers, suffix ) \136 __snd_hda_add_vmaster(codec, name, tlv, followers, suffix, true, NULL)134 unsigned int access, struct snd_kcontrol **ctl_ret); 135 #define snd_hda_add_vmaster(codec, name, tlv, followers, suffix, access) \ 136 __snd_hda_add_vmaster(codec, name, tlv, followers, suffix, true, access, NULL) 137 137 int snd_hda_codec_reset(struct hda_codec *codec); 138 138 void snd_hda_codec_register(struct hda_codec *codec); … … 140 140 141 141 #define snd_hda_regmap_sync(codec) snd_hdac_regmap_sync(&(codec)->core) 142 143 enum {144 HDA_VMUTE_OFF,145 HDA_VMUTE_ON,146 HDA_VMUTE_FOLLOW_MASTER,147 };148 142 149 143 struct hda_vmaster_mute_hook { … … 154 148 void (*hook)(void *, int); 155 149 /* below are initialized automatically */ 156 unsigned int mute_mode; /* HDA_VMUTE_XXX */157 150 struct hda_codec *codec; 158 151 }; 159 152 160 153 int snd_hda_add_vmaster_hook(struct hda_codec *codec, 161 struct hda_vmaster_mute_hook *hook, 162 bool expose_enum_ctl); 154 struct hda_vmaster_mute_hook *hook); 163 155 void snd_hda_sync_vmaster_hook(struct hda_vmaster_mute_hook *hook); 164 156 … … 181 173 * input MUX helper 182 174 */ 183 #define HDA_MAX_NUM_INPUTS 16175 #define HDA_MAX_NUM_INPUTS 36 184 176 struct hda_input_mux_item { 185 177 char label[32]; -
TabularUnified GPL/branches/uniaud32-next/alsa-kernel/pci/hda/patch_ca0132.c ¶
r693 r694 7137 7137 snd_hda_add_vmaster(codec, "Master Playback Volume", 7138 7138 spec->tlv, ca0132_alt_follower_pfxs, 7139 "Playback Volume" );7139 "Playback Volume", 0); 7140 7140 err = __snd_hda_add_vmaster(codec, "Master Playback Switch", 7141 7141 NULL, ca0132_alt_follower_pfxs, 7142 7142 "Playback Switch", 7143 true, &spec->vmaster_mute.sw_kctl);7143 true, 0, &spec->vmaster_mute.sw_kctl); 7144 7144 if (err < 0) 7145 7145 return err; -
TabularUnified GPL/branches/uniaud32-next/alsa-kernel/pci/hda/patch_cirrus.c ¶
r629 r694 10 10 #include <linux/module.h> 11 11 #include <sound/core.h> 12 #include <linux/mutex.h> 13 #include <linux/pci.h> 12 14 #include <sound/tlv.h> 13 15 #include <sound/hda_codec.h> … … 24 26 */ 25 27 28 #define CS42L42_HP_CH (2U) 29 #define CS42L42_HS_MIC_CH (1U) 30 26 31 struct cs_spec { 27 32 struct hda_gen_spec gen; … … 42 47 int (*spdif_sw_put)(struct snd_kcontrol *kcontrol, 43 48 struct snd_ctl_elem_value *ucontrol); 49 50 unsigned int cs42l42_hp_jack_in:1; 51 unsigned int cs42l42_mic_jack_in:1; 52 unsigned int cs42l42_volume_init:1; 53 char cs42l42_hp_volume[CS42L42_HP_CH]; 54 char cs42l42_hs_mic_volume[CS42L42_HS_MIC_CH]; 55 56 struct mutex cs8409_i2c_mux; 57 58 /* verb exec op override */ 59 int (*exec_verb)(struct hdac_device *dev, unsigned int cmd, 60 unsigned int flags, unsigned int *res); 44 61 }; 45 62 … … 115 132 * 1 ADC <= LineIn(sense) / MicIn / DMicIn, 116 133 * 1 SPDIF OUT => SPDIF Trasmitter(sense) 117 */134 */ 118 135 #define CS4210_DAC_NID 0x02 119 136 #define CS4210_ADC_NID 0x03 … … 134 151 { 135 152 struct cs_spec *spec = codec->spec; 153 136 154 snd_hda_codec_write(codec, spec->vendor_nid, 0, 137 155 AC_VERB_SET_COEF_INDEX, idx); … … 144 162 { 145 163 struct cs_spec *spec = codec->spec; 164 146 165 snd_hda_codec_write(codec, spec->vendor_nid, 0, 147 166 AC_VERB_SET_COEF_INDEX, idx); … … 180 199 { 181 200 unsigned int val; 201 182 202 val = snd_hda_codec_get_pincfg(codec, nid); 183 203 return (get_defcfg_connect(val) != AC_JACK_PORT_NONE); … … 198 218 * No effect if SPDIF_OUT2 is 199 219 * selected in IDX_SPDIF_CTL. 200 */220 */ 201 221 202 222 cs_vendor_coef_set(codec, IDX_BEEP_CFG, coef); … … 272 292 {0x11, AC_VERB_SET_PROC_COEF, 0x0008}, 273 293 {0x11, AC_VERB_SET_PROC_STATE, 0x00}, 274 275 #if 0 /* Don't to set to D3 as we are in power-up sequence */276 {0x07, AC_VERB_SET_POWER_STATE, 0x03}, /* S/PDIF Rx: D3 */277 {0x08, AC_VERB_SET_POWER_STATE, 0x03}, /* S/PDIF Tx: D3 */278 /*{0x01, AC_VERB_SET_POWER_STATE, 0x03},*/ /* AFG: D3 This is already handled */279 #endif280 281 294 {0} /* terminator */ 282 295 }; … … 366 379 if (spec->gen.dyn_adc_switch) { 367 380 unsigned int done = 0; 381 368 382 for (i = 0; i < spec->gen.input_mux.num_items; i++) { 369 383 int idx = spec->gen.dyn_adc_idx[i]; 384 370 385 if (done & (1 << idx)) 371 386 continue; … … 501 516 if (action == HDA_FIXUP_ACT_PRE_PROBE) { 502 517 struct cs_spec *spec = codec->spec; 518 503 519 spec->gpio_eapd_hp = 2; /* GPIO1 = headphones */ 504 520 spec->gpio_eapd_speaker = 8; /* GPIO3 = speakers */ … … 513 529 if (action == HDA_FIXUP_ACT_PRE_PROBE) { 514 530 struct cs_spec *spec = codec->spec; 531 515 532 spec->gpio_eapd_hp = 4; /* GPIO2 = headphones */ 516 533 spec->gpio_eapd_speaker = 8; /* GPIO3 = speakers */ … … 659 676 if (action == HDA_FIXUP_ACT_PRE_PROBE) { 660 677 struct cs_spec *spec = codec->spec; 678 661 679 spec->gpio_eapd_hp = 0; 662 680 spec->gpio_eapd_speaker = 1; … … 813 831 * 1 ADC <= LineIn(sense) / MicIn / DMicIn, 814 832 * 1 SPDIF OUT => SPDIF Trasmitter(sense) 815 */833 */ 816 834 817 835 /* CS4210 board names */ … … 856 874 { 857 875 struct cs_spec *spec = codec->spec; 876 858 877 if (action == HDA_FIXUP_ACT_PRE_PROBE) 859 878 spec->sense_b = 1; … … 881 900 {0x0B, AC_VERB_SET_COEF_INDEX, CS421X_IDX_DEV_CFG}, 882 901 /* 883 884 885 */902 * Disable Coefficient Index Auto-Increment(DAI)=1, 903 * PDREF=0 904 */ 886 905 {0x0B, AC_VERB_SET_PROC_COEF, 0x0001 }, 887 906 … … 970 989 coef &= ~0x0003; 971 990 coef |= (vol & 0x0003); 972 if (original_coef == coef) 973 return 0; 974 else { 991 if (original_coef != coef) { 975 992 cs_vendor_coef_set(codec, CS421X_IDX_SPK_CTL, coef); 976 993 return 1; 977 994 } 995 996 return 0; 978 997 } 979 998 … … 1014 1033 1015 1034 /* 1016 1017 */1035 * GPIO or SENSE_B forced - disconnect the DMIC pin. 1036 */ 1018 1037 def_conf = snd_hda_codec_get_pincfg(codec, CS421X_DMIC_PIN_NID); 1019 1038 def_conf &= ~AC_DEFCFG_PORT_CONN; … … 1054 1073 for (i = 0; i < cfg->dig_outs; i++) { 1055 1074 hda_nid_t nid = cfg->dig_out_pins[i]; 1075 1056 1076 if (get_wcaps(codec, nid) & AC_WCAP_UNSOL_CAP) { 1057 1077 spec->spdif_detect = 1; … … 1132 1152 #ifdef CONFIG_PM 1133 1153 /* 1134 Manage PDREF, when transitioning to D3hot1135 (DAC,ADC) -> D3, PDREF=1, AFG->D31136 */1154 * Manage PDREF, when transitioning to D3hot 1155 * (DAC,ADC) -> D3, PDREF=1, AFG->D3 1156 */ 1137 1157 static int cs421x_suspend(struct hda_codec *codec) 1138 1158 { … … 1185 1205 1186 1206 /* 1187 1188 is auto-parsed. If GPIO or SENSE_B is forced, DMIC input1189 is disabled.1190 */1207 * Update the GPIO/DMIC/SENSE_B pinmux before the configuration 1208 * is auto-parsed. If GPIO or SENSE_B is forced, DMIC input 1209 * is disabled. 1210 */ 1191 1211 cs4210_pinmux_init(codec); 1192 1212 … … 1226 1246 } 1227 1247 1248 /* Cirrus Logic CS8409 HDA bridge with 1249 * companion codec CS42L42 1250 */ 1251 #define CS8409_VENDOR_NID 0x47 1252 1253 #define CS8409_CS42L42_HP_PIN_NID 0x24 1254 #define CS8409_CS42L42_SPK_PIN_NID 0x2c 1255 #define CS8409_CS42L42_AMIC_PIN_NID 0x34 1256 #define CS8409_CS42L42_DMIC_PIN_NID 0x44 1257 #define CS8409_CS42L42_DMIC_ADC_PIN_NID 0x22 1258 1259 #define CS42L42_HSDET_AUTO_DONE 0x02 1260 #define CS42L42_HSTYPE_MASK 0x03 1261 1262 #define CS42L42_JACK_INSERTED 0x0C 1263 #define CS42L42_JACK_REMOVED 0x00 1264 1265 #define GPIO3_INT (1 << 3) 1266 #define GPIO4_INT (1 << 4) 1267 #define GPIO5_INT (1 << 5) 1268 1269 #define CS42L42_I2C_ADDR (0x48 << 1) 1270 1271 #define CIR_I2C_ADDR 0x0059 1272 #define CIR_I2C_DATA 0x005A 1273 #define CIR_I2C_CTRL 0x005B 1274 #define CIR_I2C_STATUS 0x005C 1275 #define CIR_I2C_QWRITE 0x005D 1276 #define CIR_I2C_QREAD 0x005E 1277 1278 #define CS8409_CS42L42_HP_VOL_REAL_MIN (-63) 1279 #define CS8409_CS42L42_HP_VOL_REAL_MAX (0) 1280 #define CS8409_CS42L42_AMIC_VOL_REAL_MIN (-97) 1281 #define CS8409_CS42L42_AMIC_VOL_REAL_MAX (12) 1282 #define CS8409_CS42L42_REG_HS_VOLUME_CHA (0x2301) 1283 #define CS8409_CS42L42_REG_HS_VOLUME_CHB (0x2303) 1284 #define CS8409_CS42L42_REG_AMIC_VOLUME (0x1D03) 1285 1286 struct cs8409_i2c_param { 1287 unsigned int addr; 1288 unsigned int reg; 1289 }; 1290 1291 struct cs8409_cir_param { 1292 unsigned int nid; 1293 unsigned int cir; 1294 unsigned int coeff; 1295 }; 1296 1297 enum { 1298 CS8409_BULLSEYE, 1299 CS8409_WARLOCK, 1300 CS8409_CYBORG, 1301 CS8409_FIXUPS, 1302 }; 1303 1304 static void cs8409_cs42l42_fixups(struct hda_codec *codec, 1305 const struct hda_fixup *fix, int action); 1306 static int cs8409_cs42l42_exec_verb(struct hdac_device *dev, 1307 unsigned int cmd, unsigned int flags, unsigned int *res); 1308 1309 /* Dell Inspiron models with cs8409/cs42l42 */ 1310 static const struct hda_model_fixup cs8409_models[] = { 1311 { .id = CS8409_BULLSEYE, .name = "bullseye" }, 1312 { .id = CS8409_WARLOCK, .name = "warlock" }, 1313 { .id = CS8409_CYBORG, .name = "cyborg" }, 1314 {0} 1315 }; 1316 1317 /* Dell Inspiron platforms 1318 * with cs8409 bridge and cs42l42 codec 1319 */ 1320 static const struct snd_pci_quirk cs8409_fixup_tbl[] = { 1321 SND_PCI_QUIRK(0x1028, 0x0A11, "Bullseye", CS8409_BULLSEYE), 1322 SND_PCI_QUIRK(0x1028, 0x0A12, "Bullseye", CS8409_BULLSEYE), 1323 SND_PCI_QUIRK(0x1028, 0x0A23, "Bullseye", CS8409_BULLSEYE), 1324 SND_PCI_QUIRK(0x1028, 0x0A24, "Bullseye", CS8409_BULLSEYE), 1325 SND_PCI_QUIRK(0x1028, 0x0A25, "Bullseye", CS8409_BULLSEYE), 1326 SND_PCI_QUIRK(0x1028, 0x0A29, "Bullseye", CS8409_BULLSEYE), 1327 SND_PCI_QUIRK(0x1028, 0x0A2A, "Bullseye", CS8409_BULLSEYE), 1328 SND_PCI_QUIRK(0x1028, 0x0A2B, "Bullseye", CS8409_BULLSEYE), 1329 SND_PCI_QUIRK(0x1028, 0x0AB0, "Warlock", CS8409_WARLOCK), 1330 SND_PCI_QUIRK(0x1028, 0x0AB2, "Warlock", CS8409_WARLOCK), 1331 SND_PCI_QUIRK(0x1028, 0x0AB1, "Warlock", CS8409_WARLOCK), 1332 SND_PCI_QUIRK(0x1028, 0x0AB3, "Warlock", CS8409_WARLOCK), 1333 SND_PCI_QUIRK(0x1028, 0x0AB4, "Warlock", CS8409_WARLOCK), 1334 SND_PCI_QUIRK(0x1028, 0x0AB5, "Warlock", CS8409_WARLOCK), 1335 SND_PCI_QUIRK(0x1028, 0x0AD9, "Warlock", CS8409_WARLOCK), 1336 SND_PCI_QUIRK(0x1028, 0x0ADA, "Warlock", CS8409_WARLOCK), 1337 SND_PCI_QUIRK(0x1028, 0x0ADB, "Warlock", CS8409_WARLOCK), 1338 SND_PCI_QUIRK(0x1028, 0x0ADC, "Warlock", CS8409_WARLOCK), 1339 SND_PCI_QUIRK(0x1028, 0x0AF4, "Warlock", CS8409_WARLOCK), 1340 SND_PCI_QUIRK(0x1028, 0x0AF5, "Warlock", CS8409_WARLOCK), 1341 SND_PCI_QUIRK(0x1028, 0x0A77, "Cyborg", CS8409_CYBORG), 1342 SND_PCI_QUIRK(0x1028, 0x0A78, "Cyborg", CS8409_CYBORG), 1343 SND_PCI_QUIRK(0x1028, 0x0A79, "Cyborg", CS8409_CYBORG), 1344 SND_PCI_QUIRK(0x1028, 0x0A7A, "Cyborg", CS8409_CYBORG), 1345 SND_PCI_QUIRK(0x1028, 0x0A7D, "Cyborg", CS8409_CYBORG), 1346 SND_PCI_QUIRK(0x1028, 0x0A7E, "Cyborg", CS8409_CYBORG), 1347 SND_PCI_QUIRK(0x1028, 0x0A7F, "Cyborg", CS8409_CYBORG), 1348 SND_PCI_QUIRK(0x1028, 0x0A80, "Cyborg", CS8409_CYBORG), 1349 SND_PCI_QUIRK(0x1028, 0x0ADF, "Cyborg", CS8409_CYBORG), 1350 SND_PCI_QUIRK(0x1028, 0x0AE0, "Cyborg", CS8409_CYBORG), 1351 SND_PCI_QUIRK(0x1028, 0x0AE1, "Cyborg", CS8409_CYBORG), 1352 SND_PCI_QUIRK(0x1028, 0x0AE2, "Cyborg", CS8409_CYBORG), 1353 SND_PCI_QUIRK(0x1028, 0x0AE9, "Cyborg", CS8409_CYBORG), 1354 SND_PCI_QUIRK(0x1028, 0x0AEA, "Cyborg", CS8409_CYBORG), 1355 SND_PCI_QUIRK(0x1028, 0x0AEB, "Cyborg", CS8409_CYBORG), 1356 SND_PCI_QUIRK(0x1028, 0x0AEC, "Cyborg", CS8409_CYBORG), 1357 SND_PCI_QUIRK(0x1028, 0x0AED, "Cyborg", CS8409_CYBORG), 1358 SND_PCI_QUIRK(0x1028, 0x0AEE, "Cyborg", CS8409_CYBORG), 1359 SND_PCI_QUIRK(0x1028, 0x0AEF, "Cyborg", CS8409_CYBORG), 1360 SND_PCI_QUIRK(0x1028, 0x0AF0, "Cyborg", CS8409_CYBORG), 1361 {0} /* terminator */ 1362 }; 1363 1364 static const struct hda_verb cs8409_cs42l42_init_verbs[] = { 1365 { 0x01, AC_VERB_SET_GPIO_WAKE_MASK, 0x0018 }, /* WAKE from GPIO 3,4 */ 1366 { 0x47, AC_VERB_SET_PROC_STATE, 0x0001 }, /* Enable VPW processing */ 1367 { 0x47, AC_VERB_SET_COEF_INDEX, 0x0002 }, /* Configure GPIO 6,7 */ 1368 { 0x47, AC_VERB_SET_PROC_COEF, 0x0080 }, /* I2C mode */ 1369 { 0x47, AC_VERB_SET_COEF_INDEX, 0x005b }, /* Set I2C bus speed */ 1370 { 0x47, AC_VERB_SET_PROC_COEF, 0x0200 }, /* 100kHz I2C_STO = 2 */ 1371 {0} /* terminator */ 1372 }; 1373 1374 static const struct hda_pintbl cs8409_cs42l42_pincfgs[] = { 1375 { 0x24, 0x042120f0 }, /* ASP-1-TX */ 1376 { 0x34, 0x04a12050 }, /* ASP-1-RX */ 1377 { 0x2c, 0x901000f0 }, /* ASP-2-TX */ 1378 { 0x44, 0x90a00090 }, /* DMIC-1 */ 1379 {0} /* terminator */ 1380 }; 1381 1382 static const struct hda_fixup cs8409_fixups[] = { 1383 [CS8409_BULLSEYE] = { 1384 .type = HDA_FIXUP_PINS, 1385 .v.pins = cs8409_cs42l42_pincfgs, 1386 .chained = true, 1387 .chain_id = CS8409_FIXUPS, 1388 }, 1389 [CS8409_WARLOCK] = { 1390 .type = HDA_FIXUP_PINS, 1391 .v.pins = cs8409_cs42l42_pincfgs, 1392 .chained = true, 1393 .chain_id = CS8409_FIXUPS, 1394 }, 1395 [CS8409_CYBORG] = { 1396 .type = HDA_FIXUP_PINS, 1397 .v.pins = cs8409_cs42l42_pincfgs, 1398 .chained = true, 1399 .chain_id = CS8409_FIXUPS, 1400 }, 1401 [CS8409_FIXUPS] = { 1402 .type = HDA_FIXUP_FUNC, 1403 .v.func = cs8409_cs42l42_fixups, 1404 }, 1405 }; 1406 1407 /* Vendor specific HW configuration for CS42L42 */ 1408 static const struct cs8409_i2c_param cs42l42_init_reg_seq[] = { 1409 { 0x1010, 0xB0 }, 1410 { 0x1D01, 0x00 }, 1411 { 0x1D02, 0x06 }, 1412 { 0x1D03, 0x00 }, 1413 { 0x1107, 0x01 }, 1414 { 0x1009, 0x02 }, 1415 { 0x1007, 0x03 }, 1416 { 0x1201, 0x00 }, 1417 { 0x1208, 0x13 }, 1418 { 0x1205, 0xFF }, 1419 { 0x1206, 0x00 }, 1420 { 0x1207, 0x20 }, 1421 { 0x1202, 0x0D }, 1422 { 0x2A02, 0x02 }, 1423 { 0x2A03, 0x00 }, 1424 { 0x2A04, 0x00 }, 1425 { 0x2A05, 0x02 }, 1426 { 0x2A06, 0x00 }, 1427 { 0x2A07, 0x20 }, 1428 { 0x2A08, 0x02 }, 1429 { 0x2A09, 0x00 }, 1430 { 0x2A0A, 0x80 }, 1431 { 0x2A0B, 0x02 }, 1432 { 0x2A0C, 0x00 }, 1433 { 0x2A0D, 0xA0 }, 1434 { 0x2A01, 0x0C }, 1435 { 0x2902, 0x01 }, 1436 { 0x2903, 0x02 }, 1437 { 0x2904, 0x00 }, 1438 { 0x2905, 0x00 }, 1439 { 0x2901, 0x01 }, 1440 { 0x1101, 0x0A }, 1441 { 0x1102, 0x84 }, 1442 { 0x2301, 0x00 }, 1443 { 0x2303, 0x00 }, 1444 { 0x2302, 0x3f }, 1445 { 0x2001, 0x03 }, 1446 { 0x1B75, 0xB6 }, 1447 { 0x1B73, 0xC2 }, 1448 { 0x1129, 0x01 }, 1449 { 0x1121, 0xF3 }, 1450 { 0x1103, 0x20 }, 1451 { 0x1105, 0x00 }, 1452 { 0x1112, 0xC0 }, 1453 { 0x1113, 0x80 }, 1454 { 0x1C03, 0xC0 }, 1455 { 0x1105, 0x00 }, 1456 { 0x1112, 0xC0 }, 1457 { 0x1101, 0x02 }, 1458 {0} /* Terminator */ 1459 }; 1460 1461 /* Vendor specific hw configuration for CS8409 */ 1462 static const struct cs8409_cir_param cs8409_cs42l42_hw_cfg[] = { 1463 { 0x47, 0x00, 0xb008 }, /* +PLL1/2_EN, +I2C_EN */ 1464 { 0x47, 0x01, 0x0002 }, /* ASP1/2_EN=0, ASP1_STP=1 */ 1465 { 0x47, 0x02, 0x0a80 }, /* ASP1/2_BUS_IDLE=10, +GPIO_I2C */ 1466 { 0x47, 0x19, 0x0800 }, /* ASP1.A: TX.LAP=0, TX.LSZ=24 bits, TX.LCS=0 */ 1467 { 0x47, 0x1a, 0x0820 }, /* ASP1.A: TX.RAP=0, TX.RSZ=24 bits, TX.RCS=32 */ 1468 { 0x47, 0x29, 0x0800 }, /* ASP2.A: TX.LAP=0, TX.LSZ=24 bits, TX.LCS=0 */ 1469 { 0x47, 0x2a, 0x2800 }, /* ASP2.A: TX.RAP=1, TX.RSZ=24 bits, TX.RCS=0 */ 1470 { 0x47, 0x39, 0x0800 }, /* ASP1.A: RX.LAP=0, RX.LSZ=24 bits, RX.LCS=0 */ 1471 { 0x47, 0x3a, 0x0800 }, /* ASP1.A: RX.RAP=0, RX.RSZ=24 bits, RX.RCS=0 */ 1472 { 0x47, 0x03, 0x8000 }, /* ASP1: LCHI = 00h */ 1473 { 0x47, 0x04, 0x28ff }, /* ASP1: MC/SC_SRCSEL=PLL1, LCPR=FFh */ 1474 { 0x47, 0x05, 0x0062 }, /* ASP1: MCEN=0, FSD=011, SCPOL_IN/OUT=0, SCDIV=1:4 */ 1475 { 0x47, 0x06, 0x801f }, /* ASP2: LCHI=1Fh */ 1476 { 0x47, 0x07, 0x283f }, /* ASP2: MC/SC_SRCSEL=PLL1, LCPR=3Fh */ 1477 { 0x47, 0x08, 0x805c }, /* ASP2: 5050=1, MCEN=0, FSD=010, SCPOL_IN/OUT=1, SCDIV=1:16 */ 1478 { 0x47, 0x09, 0x0023 }, /* DMIC1_MO=10b, DMIC1/2_SR=1 */ 1479 { 0x47, 0x0a, 0x0000 }, /* ASP1/2_BEEP=0 */ 1480 { 0x47, 0x01, 0x0062 }, /* ASP1/2_EN=1, ASP1_STP=1 */ 1481 { 0x47, 0x00, 0x9008 }, /* -PLL2_EN */ 1482 { 0x47, 0x68, 0x0000 }, /* TX2.A: pre-scale att.=0 dB */ 1483 { 0x47, 0x82, 0xfc03 }, /* ASP1/2_xxx_EN=1, ASP1/2_MCLK_EN=0, DMIC1_SCL_EN=1 */ 1484 { 0x47, 0xc0, 0x9999 }, /* test mode on */ 1485 { 0x47, 0xc5, 0x0000 }, /* GPIO hysteresis = 30 us */ 1486 { 0x47, 0xc0, 0x0000 }, /* test mode off */ 1487 {0} /* Terminator */ 1488 }; 1489 1490 static const struct cs8409_cir_param cs8409_cs42l42_bullseye_atn[] = { 1491 { 0x47, 0x65, 0x4000 }, /* EQ_SEL=1, EQ1/2_EN=0 */ 1492 { 0x47, 0x64, 0x4000 }, /* +EQ_ACC */ 1493 { 0x47, 0x65, 0x4010 }, /* +EQ2_EN */ 1494 { 0x47, 0x63, 0x0647 }, /* EQ_DATA_HI=0x0647 */ 1495 { 0x47, 0x64, 0xc0c7 }, /* +EQ_WRT, +EQ_ACC, EQ_ADR=0, EQ_DATA_LO=0x67 */ 1496 { 0x47, 0x63, 0x0647 }, /* EQ_DATA_HI=0x0647 */ 1497 { 0x47, 0x64, 0xc1c7 }, /* +EQ_WRT, +EQ_ACC, EQ_ADR=1, EQ_DATA_LO=0x67 */ 1498 { 0x47, 0x63, 0xf370 }, /* EQ_DATA_HI=0xf370 */ 1499 { 0x47, 0x64, 0xc271 }, /* +EQ_WRT, +EQ_ACC, EQ_ADR=2, EQ_DATA_LO=0x71 */ 1500 { 0x47, 0x63, 0x1ef8 }, /* EQ_DATA_HI=0x1ef8 */ 1501 { 0x47, 0x64, 0xc348 }, /* +EQ_WRT, +EQ_ACC, EQ_ADR=3, EQ_DATA_LO=0x48 */ 1502 { 0x47, 0x63, 0xc110 }, /* EQ_DATA_HI=0xc110 */ 1503 { 0x47, 0x64, 0xc45a }, /* +EQ_WRT, +EQ_ACC, EQ_ADR=4, EQ_DATA_LO=0x5a */ 1504 { 0x47, 0x63, 0x1f29 }, /* EQ_DATA_HI=0x1f29 */ 1505 { 0x47, 0x64, 0xc574 }, /* +EQ_WRT, +EQ_ACC, EQ_ADR=5, EQ_DATA_LO=0x74 */ 1506 { 0x47, 0x63, 0x1d7a }, /* EQ_DATA_HI=0x1d7a */ 1507 { 0x47, 0x64, 0xc653 }, /* +EQ_WRT, +EQ_ACC, EQ_ADR=6, EQ_DATA_LO=0x53 */ 1508 { 0x47, 0x63, 0xc38c }, /* EQ_DATA_HI=0xc38c */ 1509 { 0x47, 0x64, 0xc714 }, /* +EQ_WRT, +EQ_ACC, EQ_ADR=7, EQ_DATA_LO=0x14 */ 1510 { 0x47, 0x63, 0x1ca3 }, /* EQ_DATA_HI=0x1ca3 */ 1511 { 0x47, 0x64, 0xc8c7 }, /* +EQ_WRT, +EQ_ACC, EQ_ADR=8, EQ_DATA_LO=0xc7 */ 1512 { 0x47, 0x63, 0xc38c }, /* EQ_DATA_HI=0xc38c */ 1513 { 0x47, 0x64, 0xc914 }, /* +EQ_WRT, +EQ_ACC, EQ_ADR=9, EQ_DATA_LO=0x14 */ 1514 { 0x47, 0x64, 0x0000 }, /* -EQ_ACC, -EQ_WRT */ 1515 {0} /* Terminator */ 1516 }; 1517 1518 /** 1519 * cs8409_enable_i2c_clock - Enable I2C clocks 1520 * @codec: the codec instance 1521 * @enable: Enable or disable I2C clocks 1522 * 1523 * Enable or Disable I2C clocks. 1524 */ 1525 static void cs8409_enable_i2c_clock(struct hda_codec *codec, unsigned int enable) 1526 { 1527 unsigned int retval; 1528 unsigned int newval; 1529 1530 retval = cs_vendor_coef_get(codec, 0x0); 1531 newval = (enable) ? (retval | 0x8) : (retval & 0xfffffff7); 1532 cs_vendor_coef_set(codec, 0x0, newval); 1533 } 1534 1535 /** 1536 * cs8409_i2c_wait_complete - Wait for I2C transaction 1537 * @codec: the codec instance 1538 * 1539 * Wait for I2C transaction to complete. 1540 * Return -1 if transaction wait times out. 1541 */ 1542 static int cs8409_i2c_wait_complete(struct hda_codec *codec) 1543 { 1544 int repeat = 5; 1545 unsigned int retval; 1546 1547 do { 1548 retval = cs_vendor_coef_get(codec, CIR_I2C_STATUS); 1549 if ((retval & 0x18) != 0x18) { 1550 usleep_range(2000, 4000); 1551 --repeat; 1552 } else 1553 return 0; 1554 1555 } while (repeat); 1556 1557 return -1; 1558 } 1559 1560 /** 1561 * cs8409_i2c_read - CS8409 I2C Read. 1562 * @codec: the codec instance 1563 * @i2c_address: I2C Address 1564 * @i2c_reg: Register to read 1565 * @paged: Is a paged transaction 1566 * 1567 * CS8409 I2C Read. 1568 * Returns negative on error, otherwise returns read value in bits 0-7. 1569 */ 1570 static int cs8409_i2c_read(struct hda_codec *codec, 1571 unsigned int i2c_address, 1572 unsigned int i2c_reg, 1573 unsigned int paged) 1574 { 1575 unsigned int i2c_reg_data; 1576 unsigned int read_data; 1577 1578 cs8409_enable_i2c_clock(codec, 1); 1579 cs_vendor_coef_set(codec, CIR_I2C_ADDR, i2c_address); 1580 1581 if (paged) { 1582 cs_vendor_coef_set(codec, CIR_I2C_QWRITE, i2c_reg >> 8); 1583 if (cs8409_i2c_wait_complete(codec) < 0) { 1584 codec_err(codec, 1585 "%s() Paged Transaction Failed 0x%02x : 0x%04x\n", 1586 __func__, i2c_address, i2c_reg); 1587 return -EIO; 1588 } 1589 } 1590 1591 i2c_reg_data = (i2c_reg << 8) & 0x0ffff; 1592 cs_vendor_coef_set(codec, CIR_I2C_QREAD, i2c_reg_data); 1593 if (cs8409_i2c_wait_complete(codec) < 0) { 1594 codec_err(codec, "%s() Transaction Failed 0x%02x : 0x%04x\n", 1595 __func__, i2c_address, i2c_reg); 1596 return -EIO; 1597 } 1598 1599 /* Register in bits 15-8 and the data in 7-0 */ 1600 read_data = cs_vendor_coef_get(codec, CIR_I2C_QREAD); 1601 1602 cs8409_enable_i2c_clock(codec, 0); 1603 1604 return read_data & 0x0ff; 1605 } 1606 1607 /** 1608 * cs8409_i2c_write - CS8409 I2C Write. 1609 * @codec: the codec instance 1610 * @i2c_address: I2C Address 1611 * @i2c_reg: Register to write to 1612 * @i2c_data: Data to write 1613 * @paged: Is a paged transaction 1614 * 1615 * CS8409 I2C Write. 1616 * Returns negative on error, otherwise returns 0. 1617 */ 1618 static int cs8409_i2c_write(struct hda_codec *codec, 1619 unsigned int i2c_address, unsigned int i2c_reg, 1620 unsigned int i2c_data, 1621 unsigned int paged) 1622 { 1623 unsigned int i2c_reg_data; 1624 1625 cs8409_enable_i2c_clock(codec, 1); 1626 cs_vendor_coef_set(codec, CIR_I2C_ADDR, i2c_address); 1627 1628 if (paged) { 1629 cs_vendor_coef_set(codec, CIR_I2C_QWRITE, i2c_reg >> 8); 1630 if (cs8409_i2c_wait_complete(codec) < 0) { 1631 codec_err(codec, 1632 "%s() Paged Transaction Failed 0x%02x : 0x%04x\n", 1633 __func__, i2c_address, i2c_reg); 1634 return -EIO; 1635 } 1636 } 1637 1638 i2c_reg_data = ((i2c_reg << 8) & 0x0ff00) | (i2c_data & 0x0ff); 1639 cs_vendor_coef_set(codec, CIR_I2C_QWRITE, i2c_reg_data); 1640 1641 if (cs8409_i2c_wait_complete(codec) < 0) { 1642 codec_err(codec, "%s() Transaction Failed 0x%02x : 0x%04x\n", 1643 __func__, i2c_address, i2c_reg); 1644 return -EIO; 1645 } 1646 1647 cs8409_enable_i2c_clock(codec, 0); 1648 1649 return 0; 1650 } 1651 1652 static int cs8409_cs42l42_volume_info(struct snd_kcontrol *kcontrol, 1653 struct snd_ctl_elem_info *uinfo) 1654 { 1655 struct hda_codec *codec = snd_kcontrol_chip(kcontrol); 1656 u16 nid = get_amp_nid(kcontrol); 1657 u8 chs = get_amp_channels(kcontrol); 1658 1659 codec_dbg(codec, "%s() nid: %d\n", __func__, nid); 1660 switch (nid) { 1661 case CS8409_CS42L42_HP_PIN_NID: 1662 uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER; 1663 uinfo->count = chs == 3 ? 2 : 1; 1664 uinfo->value.integer.min = CS8409_CS42L42_HP_VOL_REAL_MIN; 1665 uinfo->value.integer.max = CS8409_CS42L42_HP_VOL_REAL_MAX; 1666 break; 1667 case CS8409_CS42L42_AMIC_PIN_NID: 1668 uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER; 1669 uinfo->count = chs == 3 ? 2 : 1; 1670 uinfo->value.integer.min = CS8409_CS42L42_AMIC_VOL_REAL_MIN; 1671 uinfo->value.integer.max = CS8409_CS42L42_AMIC_VOL_REAL_MAX; 1672 break; 1673 default: 1674 break; 1675 } 1676 return 0; 1677 } 1678 1679 static void cs8409_cs42l42_update_volume(struct hda_codec *codec) 1680 { 1681 struct cs_spec *spec = codec->spec; 1682 int data; 1683 1684 mutex_lock(&spec->cs8409_i2c_mux); 1685 data = cs8409_i2c_read(codec, CS42L42_I2C_ADDR, 1686 CS8409_CS42L42_REG_HS_VOLUME_CHA, 1); 1687 if (data >= 0) 1688 spec->cs42l42_hp_volume[0] = -data; 1689 else 1690 spec->cs42l42_hp_volume[0] = CS8409_CS42L42_HP_VOL_REAL_MIN; 1691 data = cs8409_i2c_read(codec, CS42L42_I2C_ADDR, 1692 CS8409_CS42L42_REG_HS_VOLUME_CHB, 1); 1693 if (data >= 0) 1694 spec->cs42l42_hp_volume[1] = -data; 1695 else 1696 spec->cs42l42_hp_volume[1] = CS8409_CS42L42_HP_VOL_REAL_MIN; 1697 data = cs8409_i2c_read(codec, CS42L42_I2C_ADDR, 1698 CS8409_CS42L42_REG_AMIC_VOLUME, 1); 1699 if (data >= 0) 1700 spec->cs42l42_hs_mic_volume[0] = -data; 1701 else 1702 spec->cs42l42_hs_mic_volume[0] = CS8409_CS42L42_AMIC_VOL_REAL_MIN; 1703 mutex_unlock(&spec->cs8409_i2c_mux); 1704 spec->cs42l42_volume_init = 1; 1705 } 1706 1707 static int cs8409_cs42l42_volume_get(struct snd_kcontrol *kcontrol, 1708 struct snd_ctl_elem_value *ucontrol) 1709 { 1710 struct hda_codec *codec = snd_kcontrol_chip(kcontrol); 1711 struct cs_spec *spec = codec->spec; 1712 hda_nid_t nid = get_amp_nid(kcontrol); 1713 int chs = get_amp_channels(kcontrol); 1714 long *valp = ucontrol->value.integer.value; 1715 1716 if (!spec->cs42l42_volume_init) { 1717 snd_hda_power_up(codec); 1718 cs8409_cs42l42_update_volume(codec); 1719 snd_hda_power_down(codec); 1720 } 1721 switch (nid) { 1722 case CS8409_CS42L42_HP_PIN_NID: 1723 if (chs & BIT(0)) 1724 *valp++ = spec->cs42l42_hp_volume[0]; 1725 if (chs & BIT(1)) 1726 *valp++ = spec->cs42l42_hp_volume[1]; 1727 break; 1728 case CS8409_CS42L42_AMIC_PIN_NID: 1729 if (chs & BIT(0)) 1730 *valp++ = spec->cs42l42_hs_mic_volume[0]; 1731 break; 1732 default: 1733 break; 1734 } 1735 return 0; 1736 } 1737 1738 static int cs8409_cs42l42_volume_put(struct snd_kcontrol *kcontrol, 1739 struct snd_ctl_elem_value *ucontrol) 1740 { 1741 struct hda_codec *codec = snd_kcontrol_chip(kcontrol); 1742 struct cs_spec *spec = codec->spec; 1743 hda_nid_t nid = get_amp_nid(kcontrol); 1744 int chs = get_amp_channels(kcontrol); 1745 long *valp = ucontrol->value.integer.value; 1746 int change = 0; 1747 char vol; 1748 1749 snd_hda_power_up(codec); 1750 switch (nid) { 1751 case CS8409_CS42L42_HP_PIN_NID: 1752 mutex_lock(&spec->cs8409_i2c_mux); 1753 if (chs & BIT(0)) { 1754 vol = -(*valp); 1755 change = cs8409_i2c_write(codec, CS42L42_I2C_ADDR, 1756 CS8409_CS42L42_REG_HS_VOLUME_CHA, vol, 1); 1757 valp++; 1758 } 1759 if (chs & BIT(1)) { 1760 vol = -(*valp); 1761 change |= cs8409_i2c_write(codec, CS42L42_I2C_ADDR, 1762 CS8409_CS42L42_REG_HS_VOLUME_CHB, vol, 1); 1763 } 1764 mutex_unlock(&spec->cs8409_i2c_mux); 1765 break; 1766 case CS8409_CS42L42_AMIC_PIN_NID: 1767 mutex_lock(&spec->cs8409_i2c_mux); 1768 if (chs & BIT(0)) { 1769 change = cs8409_i2c_write( 1770 codec, CS42L42_I2C_ADDR, 1771 CS8409_CS42L42_REG_AMIC_VOLUME, (char)*valp, 1); 1772 valp++; 1773 } 1774 mutex_unlock(&spec->cs8409_i2c_mux); 1775 break; 1776 default: 1777 break; 1778 } 1779 cs8409_cs42l42_update_volume(codec); 1780 snd_hda_power_down(codec); 1781 return change; 1782 } 1783 1784 static const DECLARE_TLV_DB_SCALE( 1785 cs8409_cs42l42_hp_db_scale, 1786 CS8409_CS42L42_HP_VOL_REAL_MIN * 100, 100, 1); 1787 1788 static const DECLARE_TLV_DB_SCALE( 1789 cs8409_cs42l42_amic_db_scale, 1790 CS8409_CS42L42_AMIC_VOL_REAL_MIN * 100, 100, 1); 1791 1792 static const struct snd_kcontrol_new cs8409_cs42l42_hp_volume_mixer = { 1793 .iface = SNDRV_CTL_ELEM_IFACE_MIXER, 1794 .index = 0, 1795 .name = "Headphone Playback Volume", 1796 .subdevice = (HDA_SUBDEV_AMP_FLAG | HDA_SUBDEV_NID_FLAG), 1797 .access = (SNDRV_CTL_ELEM_ACCESS_READWRITE 1798 | SNDRV_CTL_ELEM_ACCESS_TLV_READ), 1799 .info = cs8409_cs42l42_volume_info, 1800 .get = cs8409_cs42l42_volume_get, 1801 .put = cs8409_cs42l42_volume_put, 1802 .tlv = { .p = cs8409_cs42l42_hp_db_scale }, 1803 .private_value = HDA_COMPOSE_AMP_VAL( 1804 CS8409_CS42L42_HP_PIN_NID, 3, 0, HDA_OUTPUT) 1805 | HDA_AMP_VAL_MIN_MUTE 1806 }; 1807 1808 static const struct snd_kcontrol_new cs8409_cs42l42_amic_volume_mixer = { 1809 .iface = SNDRV_CTL_ELEM_IFACE_MIXER, 1810 .index = 0, 1811 .name = "Mic Capture Volume", 1812 .subdevice = (HDA_SUBDEV_AMP_FLAG | HDA_SUBDEV_NID_FLAG), 1813 .access = (SNDRV_CTL_ELEM_ACCESS_READWRITE 1814 | SNDRV_CTL_ELEM_ACCESS_TLV_READ), 1815 .info = cs8409_cs42l42_volume_info, 1816 .get = cs8409_cs42l42_volume_get, 1817 .put = cs8409_cs42l42_volume_put, 1818 .tlv = { .p = cs8409_cs42l42_amic_db_scale }, 1819 .private_value = HDA_COMPOSE_AMP_VAL( 1820 CS8409_CS42L42_AMIC_PIN_NID, 1, 0, HDA_INPUT) 1821 | HDA_AMP_VAL_MIN_MUTE 1822 }; 1823 1824 /* Assert/release RTS# line to CS42L42 */ 1825 static void cs8409_cs42l42_reset(struct hda_codec *codec) 1826 { 1827 struct cs_spec *spec = codec->spec; 1828 1829 /* Assert RTS# line */ 1830 snd_hda_codec_write(codec, 1831 codec->core.afg, 0, AC_VERB_SET_GPIO_DATA, 0); 1832 /* wait ~10ms */ 1833 usleep_range(10000, 15000); 1834 /* Release RTS# line */ 1835 snd_hda_codec_write(codec, 1836 codec->core.afg, 0, AC_VERB_SET_GPIO_DATA, GPIO5_INT); 1837 /* wait ~10ms */ 1838 usleep_range(10000, 15000); 1839 1840 mutex_lock(&spec->cs8409_i2c_mux); 1841 1842 /* Clear interrupts, by reading interrupt status registers */ 1843 cs8409_i2c_read(codec, CS42L42_I2C_ADDR, 0x1308, 1); 1844 cs8409_i2c_read(codec, CS42L42_I2C_ADDR, 0x1309, 1); 1845 cs8409_i2c_read(codec, CS42L42_I2C_ADDR, 0x130A, 1); 1846 cs8409_i2c_read(codec, CS42L42_I2C_ADDR, 0x130F, 1); 1847 1848 mutex_unlock(&spec->cs8409_i2c_mux); 1849 1850 } 1851 1852 /* Configure CS42L42 slave codec for jack autodetect */ 1853 static void cs8409_cs42l42_enable_jack_detect(struct hda_codec *codec) 1854 { 1855 struct cs_spec *spec = codec->spec; 1856 1857 mutex_lock(&spec->cs8409_i2c_mux); 1858 1859 /* Set TIP_SENSE_EN for analog front-end of tip sense. */ 1860 cs8409_i2c_write(codec, CS42L42_I2C_ADDR, 0x1b70, 0x0020, 1); 1861 /* Clear WAKE# */ 1862 cs8409_i2c_write(codec, CS42L42_I2C_ADDR, 0x1b71, 0x0001, 1); 1863 /* Wait ~2.5ms */ 1864 usleep_range(2500, 3000); 1865 /* Set mode WAKE# output follows the combination logic directly */ 1866 cs8409_i2c_write(codec, CS42L42_I2C_ADDR, 0x1b71, 0x0020, 1); 1867 /* Clear interrupts status */ 1868 cs8409_i2c_read(codec, CS42L42_I2C_ADDR, 0x130f, 1); 1869 cs8409_i2c_read(codec, CS42L42_I2C_ADDR, 0x1b7b, 1); 1870 /* Enable interrupt */ 1871 cs8409_i2c_write(codec, CS42L42_I2C_ADDR, 0x1320, 0x03, 1); 1872 cs8409_i2c_write(codec, CS42L42_I2C_ADDR, 0x1b79, 0x00, 1); 1873 1874 mutex_unlock(&spec->cs8409_i2c_mux); 1875 } 1876 1877 /* Enable and run CS42L42 slave codec jack auto detect */ 1878 static void cs8409_cs42l42_run_jack_detect(struct hda_codec *codec) 1879 { 1880 struct cs_spec *spec = codec->spec; 1881 1882 mutex_lock(&spec->cs8409_i2c_mux); 1883 1884 /* Clear interrupts */ 1885 cs8409_i2c_read(codec, CS42L42_I2C_ADDR, 0x1308, 1); 1886 cs8409_i2c_read(codec, CS42L42_I2C_ADDR, 0x1b77, 1); 1887 1888 cs8409_i2c_write(codec, CS42L42_I2C_ADDR, 0x1102, 0x87, 1); 1889 cs8409_i2c_write(codec, CS42L42_I2C_ADDR, 0x1f06, 0x86, 1); 1890 cs8409_i2c_write(codec, CS42L42_I2C_ADDR, 0x1b74, 0x07, 1); 1891 cs8409_i2c_write(codec, CS42L42_I2C_ADDR, 0x131b, 0x01, 1); 1892 cs8409_i2c_write(codec, CS42L42_I2C_ADDR, 0x1120, 0x80, 1); 1893 /* Wait ~110ms*/ 1894 usleep_range(110000, 200000); 1895 cs8409_i2c_write(codec, CS42L42_I2C_ADDR, 0x111f, 0x77, 1); 1896 cs8409_i2c_write(codec, CS42L42_I2C_ADDR, 0x1120, 0xc0, 1); 1897 /* Wait ~10ms */ 1898 usleep_range(10000, 25000); 1899 1900 mutex_unlock(&spec->cs8409_i2c_mux); 1901 1902 } 1903 1904 static void cs8409_cs42l42_reg_setup(struct hda_codec *codec) 1905 { 1906 const struct cs8409_i2c_param *seq = cs42l42_init_reg_seq; 1907 struct cs_spec *spec = codec->spec; 1908 1909 mutex_lock(&spec->cs8409_i2c_mux); 1910 1911 for (; seq->addr; seq++) 1912 cs8409_i2c_write(codec, CS42L42_I2C_ADDR, seq->addr, seq->reg, 1); 1913 1914 mutex_unlock(&spec->cs8409_i2c_mux); 1915 1916 } 1917 1918 /* 1919 * In the case of CS8409 we do not have unsolicited events from NID's 0x24 1920 * and 0x34 where hs mic and hp are connected. Companion codec CS42L42 will 1921 * generate interrupt via gpio 4 to notify jack events. We have to overwrite 1922 * generic snd_hda_jack_unsol_event(), read CS42L42 jack detect status registers 1923 * and then notify status via generic snd_hda_jack_unsol_event() call. 1924 */ 1925 static void cs8409_jack_unsol_event(struct hda_codec *codec, unsigned int res) 1926 { 1927 struct cs_spec *spec = codec->spec; 1928 int status_changed = 0; 1929 int reg_cdc_status; 1930 int reg_hs_status; 1931 int reg_ts_status; 1932 int type; 1933 struct hda_jack_tbl *jk; 1934 1935 /* jack_unsol_event() will be called every time gpio line changing state. 1936 * In this case gpio4 line goes up as a result of reading interrupt status 1937 * registers in previous cs8409_jack_unsol_event() call. 1938 * We don't need to handle this event, ignoring... 1939 */ 1940 if ((res & (1 << 4))) 1941 return; 1942 1943 mutex_lock(&spec->cs8409_i2c_mux); 1944 1945 /* Read jack detect status registers */ 1946 reg_cdc_status = cs8409_i2c_read(codec, CS42L42_I2C_ADDR, 0x1308, 1); 1947 reg_hs_status = cs8409_i2c_read(codec, CS42L42_I2C_ADDR, 0x1124, 1); 1948 reg_ts_status = cs8409_i2c_read(codec, CS42L42_I2C_ADDR, 0x130f, 1); 1949 1950 /* Clear interrupts, by reading interrupt status registers */ 1951 cs8409_i2c_read(codec, CS42L42_I2C_ADDR, 0x1b7b, 1); 1952 1953 mutex_unlock(&spec->cs8409_i2c_mux); 1954 1955 /* If status values are < 0, read error has occurred. */ 1956 if (reg_cdc_status < 0 || reg_hs_status < 0 || reg_ts_status < 0) 1957 return; 1958 1959 /* HSDET_AUTO_DONE */ 1960 if (reg_cdc_status & CS42L42_HSDET_AUTO_DONE) { 1961 1962 type = ((reg_hs_status & CS42L42_HSTYPE_MASK) + 1); 1963 /* CS42L42 reports optical jack as type 4 1964 * We don't handle optical jack 1965 */ 1966 if (type != 4) { 1967 if (!spec->cs42l42_hp_jack_in) { 1968 status_changed = 1; 1969 spec->cs42l42_hp_jack_in = 1; 1970 } 1971 /* type = 3 has no mic */ 1972 if ((!spec->cs42l42_mic_jack_in) && (type != 3)) { 1973 status_changed = 1; 1974 spec->cs42l42_mic_jack_in = 1; 1975 } 1976 } else { 1977 if (spec->cs42l42_hp_jack_in || spec->cs42l42_mic_jack_in) { 1978 status_changed = 1; 1979 spec->cs42l42_hp_jack_in = 0; 1980 spec->cs42l42_mic_jack_in = 0; 1981 } 1982 } 1983 1984 } else { 1985 /* TIP_SENSE INSERT/REMOVE */ 1986 switch (reg_ts_status) { 1987 case CS42L42_JACK_INSERTED: 1988 cs8409_cs42l42_run_jack_detect(codec); 1989 break; 1990 1991 case CS42L42_JACK_REMOVED: 1992 if (spec->cs42l42_hp_jack_in || spec->cs42l42_mic_jack_in) { 1993 status_changed = 1; 1994 spec->cs42l42_hp_jack_in = 0; 1995 spec->cs42l42_mic_jack_in = 0; 1996 } 1997 break; 1998 1999 default: 2000 /* jack in transition */ 2001 status_changed = 0; 2002 break; 2003 } 2004 } 2005 2006 if (status_changed) { 2007 2008 snd_hda_set_pin_ctl(codec, CS8409_CS42L42_SPK_PIN_NID, 2009 spec->cs42l42_hp_jack_in ? 0 : PIN_OUT); 2010 2011 /* Report jack*/ 2012 jk = snd_hda_jack_tbl_get_mst(codec, CS8409_CS42L42_HP_PIN_NID, 0); 2013 if (jk) { 2014 snd_hda_jack_unsol_event(codec, 2015 (jk->tag << AC_UNSOL_RES_TAG_SHIFT) & AC_UNSOL_RES_TAG); 2016 } 2017 /* Report jack*/ 2018 jk = snd_hda_jack_tbl_get_mst(codec, CS8409_CS42L42_AMIC_PIN_NID, 0); 2019 if (jk) { 2020 snd_hda_jack_unsol_event(codec, 2021 (jk->tag << AC_UNSOL_RES_TAG_SHIFT) & AC_UNSOL_RES_TAG); 2022 } 2023 } 2024 } 2025 2026 #ifdef CONFIG_PM 2027 /* Manage PDREF, when transition to D3hot */ 2028 static int cs8409_suspend(struct hda_codec *codec) 2029 { 2030 struct cs_spec *spec = codec->spec; 2031 2032 mutex_lock(&spec->cs8409_i2c_mux); 2033 /* Power down CS42L42 ASP/EQ/MIX/HP */ 2034 cs8409_i2c_write(codec, CS42L42_I2C_ADDR, 0x1101, 0xfe, 1); 2035 mutex_unlock(&spec->cs8409_i2c_mux); 2036 /* Assert CS42L42 RTS# line */ 2037 snd_hda_codec_write(codec, 2038 codec->core.afg, 0, AC_VERB_SET_GPIO_DATA, 0); 2039 2040 snd_hda_shutup_pins(codec); 2041 2042 return 0; 2043 } 2044 #endif 2045 2046 /* Enable/Disable Unsolicited Response for gpio(s) 3,4 */ 2047 static void cs8409_enable_ur(struct hda_codec *codec, int flag) 2048 { 2049 /* GPIO4 INT# and GPIO3 WAKE# */ 2050 snd_hda_codec_write(codec, codec->core.afg, 2051 0, AC_VERB_SET_GPIO_UNSOLICITED_RSP_MASK, 2052 flag ? (GPIO3_INT | GPIO4_INT) : 0); 2053 2054 snd_hda_codec_write(codec, codec->core.afg, 2055 0, AC_VERB_SET_UNSOLICITED_ENABLE, 2056 flag ? AC_UNSOL_ENABLED : 0); 2057 2058 } 2059 2060 /* Vendor specific HW configuration 2061 * PLL, ASP, I2C, SPI, GPIOs, DMIC etc... 2062 */ 2063 static void cs8409_cs42l42_hw_init(struct hda_codec *codec) 2064 { 2065 const struct cs8409_cir_param *seq = cs8409_cs42l42_hw_cfg; 2066 const struct cs8409_cir_param *seq_bullseye = cs8409_cs42l42_bullseye_atn; 2067 struct cs_spec *spec = codec->spec; 2068 2069 if (spec->gpio_mask) { 2070 snd_hda_codec_write(codec, 0x01, 0, AC_VERB_SET_GPIO_MASK, 2071 spec->gpio_mask); 2072 snd_hda_codec_write(codec, 0x01, 0, AC_VERB_SET_GPIO_DIRECTION, 2073 spec->gpio_dir); 2074 snd_hda_codec_write(codec, 0x01, 0, AC_VERB_SET_GPIO_DATA, 2075 spec->gpio_data); 2076 } 2077 2078 for (; seq->nid; seq++) 2079 cs_vendor_coef_set(codec, seq->cir, seq->coeff); 2080 2081 if (codec->fixup_id == CS8409_BULLSEYE) 2082 for (; seq_bullseye->nid; seq_bullseye++) 2083 cs_vendor_coef_set(codec, seq_bullseye->cir, seq_bullseye->coeff); 2084 2085 /* Disable Unsolicited Response during boot */ 2086 cs8409_enable_ur(codec, 0); 2087 2088 /* Reset CS42L42 */ 2089 cs8409_cs42l42_reset(codec); 2090 2091 /* Initialise CS42L42 companion codec */ 2092 cs8409_cs42l42_reg_setup(codec); 2093 2094 if (codec->fixup_id == CS8409_WARLOCK || 2095 codec->fixup_id == CS8409_CYBORG) { 2096 /* FULL_SCALE_VOL = 0 for Warlock / Cyborg */ 2097 mutex_lock(&spec->cs8409_i2c_mux); 2098 cs8409_i2c_write(codec, CS42L42_I2C_ADDR, 0x2001, 0x01, 1); 2099 mutex_unlock(&spec->cs8409_i2c_mux); 2100 /* DMIC1_MO=00b, DMIC1/2_SR=1 */ 2101 cs_vendor_coef_set(codec, 0x09, 0x0003); 2102 } 2103 2104 /* Restore Volumes after Resume */ 2105 if (spec->cs42l42_volume_init) { 2106 mutex_lock(&spec->cs8409_i2c_mux); 2107 cs8409_i2c_write(codec, CS42L42_I2C_ADDR, 2108 CS8409_CS42L42_REG_HS_VOLUME_CHA, 2109 -spec->cs42l42_hp_volume[0], 2110 1); 2111 cs8409_i2c_write(codec, CS42L42_I2C_ADDR, 2112 CS8409_CS42L42_REG_HS_VOLUME_CHB, 2113 -spec->cs42l42_hp_volume[1], 2114 1); 2115 cs8409_i2c_write(codec, CS42L42_I2C_ADDR, 2116 CS8409_CS42L42_REG_AMIC_VOLUME, 2117 spec->cs42l42_hs_mic_volume[0], 2118 1); 2119 mutex_unlock(&spec->cs8409_i2c_mux); 2120 } 2121 2122 cs8409_cs42l42_update_volume(codec); 2123 2124 cs8409_cs42l42_enable_jack_detect(codec); 2125 2126 /* Enable Unsolicited Response */ 2127 cs8409_enable_ur(codec, 1); 2128 } 2129 2130 static int cs8409_cs42l42_init(struct hda_codec *codec) 2131 { 2132 int ret = snd_hda_gen_init(codec); 2133 2134 if (!ret) 2135 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_INIT); 2136 2137 return ret; 2138 } 2139 2140 static const struct hda_codec_ops cs8409_cs42l42_patch_ops = { 2141 .build_controls = cs_build_controls, 2142 .build_pcms = snd_hda_gen_build_pcms, 2143 .init = cs8409_cs42l42_init, 2144 .free = cs_free, 2145 .unsol_event = cs8409_jack_unsol_event, 2146 #ifdef CONFIG_PM 2147 .suspend = cs8409_suspend, 2148 #endif 2149 }; 2150 2151 static void cs8409_cs42l42_fixups(struct hda_codec *codec, 2152 const struct hda_fixup *fix, int action) 2153 { 2154 struct cs_spec *spec = codec->spec; 2155 int caps; 2156 2157 switch (action) { 2158 case HDA_FIXUP_ACT_PRE_PROBE: 2159 snd_hda_add_verbs(codec, cs8409_cs42l42_init_verbs); 2160 /* verb exec op override */ 2161 spec->exec_verb = codec->core.exec_verb; 2162 codec->core.exec_verb = cs8409_cs42l42_exec_verb; 2163 2164 mutex_init(&spec->cs8409_i2c_mux); 2165 2166 codec->patch_ops = cs8409_cs42l42_patch_ops; 2167 2168 spec->gen.suppress_auto_mute = 1; 2169 spec->gen.no_primary_hp = 1; 2170 spec->gen.suppress_vmaster = 1; 2171 2172 /* GPIO 5 out, 3,4 in */ 2173 spec->gpio_dir = GPIO5_INT; 2174 spec->gpio_data = 0; 2175 spec->gpio_mask = 0x03f; 2176 2177 spec->cs42l42_hp_jack_in = 0; 2178 spec->cs42l42_mic_jack_in = 0; 2179 2180 /* Basic initial sequence for specific hw configuration */ 2181 snd_hda_sequence_write(codec, cs8409_cs42l42_init_verbs); 2182 2183 /* CS8409 is simple HDA bridge and intended to be used with a remote 2184 * companion codec. Most of input/output PIN(s) have only basic 2185 * capabilities. NID(s) 0x24 and 0x34 have only OUTC and INC 2186 * capabilities and no presence detect capable (PDC) and call to 2187 * snd_hda_gen_build_controls() will mark them as non detectable 2188 * phantom jacks. However, in this configuration companion codec 2189 * CS42L42 is connected to these pins and it has jack detect 2190 * capabilities. We have to override pin capabilities, 2191 * otherwise they will not be created as input devices. 2192 */ 2193 caps = snd_hdac_read_parm(&codec->core, CS8409_CS42L42_HP_PIN_NID, 2194 AC_PAR_PIN_CAP); 2195 if (caps >= 0) 2196 snd_hdac_override_parm(&codec->core, 2197 CS8409_CS42L42_HP_PIN_NID, AC_PAR_PIN_CAP, 2198 (caps | (AC_PINCAP_IMP_SENSE | AC_PINCAP_PRES_DETECT))); 2199 2200 caps = snd_hdac_read_parm(&codec->core, CS8409_CS42L42_AMIC_PIN_NID, 2201 AC_PAR_PIN_CAP); 2202 if (caps >= 0) 2203 snd_hdac_override_parm(&codec->core, 2204 CS8409_CS42L42_AMIC_PIN_NID, AC_PAR_PIN_CAP, 2205 (caps | (AC_PINCAP_IMP_SENSE | AC_PINCAP_PRES_DETECT))); 2206 2207 snd_hda_override_wcaps(codec, CS8409_CS42L42_HP_PIN_NID, 2208 (get_wcaps(codec, CS8409_CS42L42_HP_PIN_NID) | AC_WCAP_UNSOL_CAP)); 2209 2210 snd_hda_override_wcaps(codec, CS8409_CS42L42_AMIC_PIN_NID, 2211 (get_wcaps(codec, CS8409_CS42L42_AMIC_PIN_NID) | AC_WCAP_UNSOL_CAP)); 2212 break; 2213 case HDA_FIXUP_ACT_PROBE: 2214 2215 /* Set initial DMIC volume to -26 dB */ 2216 snd_hda_codec_amp_init_stereo(codec, CS8409_CS42L42_DMIC_ADC_PIN_NID, 2217 HDA_INPUT, 0, 0xff, 0x19); 2218 snd_hda_gen_add_kctl(&spec->gen, 2219 NULL, &cs8409_cs42l42_hp_volume_mixer); 2220 snd_hda_gen_add_kctl(&spec->gen, 2221 NULL, &cs8409_cs42l42_amic_volume_mixer); 2222 cs8409_cs42l42_hw_init(codec); 2223 snd_hda_codec_set_name(codec, "CS8409/CS42L42"); 2224 break; 2225 case HDA_FIXUP_ACT_INIT: 2226 cs8409_cs42l42_hw_init(codec); 2227 fallthrough; 2228 case HDA_FIXUP_ACT_BUILD: 2229 /* Run jack auto detect first time on boot 2230 * after controls have been added, to check if jack has 2231 * been already plugged in. 2232 * Run immediately after init. 2233 */ 2234 cs8409_cs42l42_run_jack_detect(codec); 2235 usleep_range(100000, 150000); 2236 break; 2237 default: 2238 break; 2239 } 2240 } 2241 2242 static int cs8409_cs42l42_exec_verb(struct hdac_device *dev, 2243 unsigned int cmd, unsigned int flags, unsigned int *res) 2244 { 2245 struct hda_codec *codec = container_of(dev, struct hda_codec, core); 2246 struct cs_spec *spec = codec->spec; 2247 2248 unsigned int nid = ((cmd >> 20) & 0x07f); 2249 unsigned int verb = ((cmd >> 8) & 0x0fff); 2250 2251 /* CS8409 pins have no AC_PINSENSE_PRESENCE 2252 * capabilities. We have to intercept 2 calls for pins 0x24 and 0x34 2253 * and return correct pin sense values for read_pin_sense() call from 2254 * hda_jack based on CS42L42 jack detect status. 2255 */ 2256 switch (nid) { 2257 case CS8409_CS42L42_HP_PIN_NID: 2258 if (verb == AC_VERB_GET_PIN_SENSE) { 2259 *res = (spec->cs42l42_hp_jack_in) ? AC_PINSENSE_PRESENCE : 0; 2260 return 0; 2261 } 2262 break; 2263 2264 case CS8409_CS42L42_AMIC_PIN_NID: 2265 if (verb == AC_VERB_GET_PIN_SENSE) { 2266 *res = (spec->cs42l42_mic_jack_in) ? AC_PINSENSE_PRESENCE : 0; 2267 return 0; 2268 } 2269 break; 2270 2271 default: 2272 break; 2273 } 2274 2275 return spec->exec_verb(dev, cmd, flags, res); 2276 } 2277 2278 static int patch_cs8409(struct hda_codec *codec) 2279 { 2280 int err; 2281 2282 if (!cs_alloc_spec(codec, CS8409_VENDOR_NID)) 2283 return -ENOMEM; 2284 2285 snd_hda_pick_fixup(codec, 2286 cs8409_models, cs8409_fixup_tbl, cs8409_fixups); 2287 2288 codec_dbg(codec, "Picked ID=%d, VID=%08x, DEV=%08x\n", 2289 codec->fixup_id, 2290 codec->bus->pci->subsystem_vendor, 2291 codec->bus->pci->subsystem_device); 2292 2293 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PRE_PROBE); 2294 2295 err = cs_parse_auto_config(codec); 2296 if (err < 0) { 2297 cs_free(codec); 2298 return err; 2299 } 2300 2301 snd_hda_apply_fixup(codec, HDA_FIXUP_ACT_PROBE); 2302 return 0; 2303 } 1228 2304 1229 2305 /* … … 1236 2312 HDA_CODEC_ENTRY(0x10134210, "CS4210", patch_cs4210), 1237 2313 HDA_CODEC_ENTRY(0x10134213, "CS4213", patch_cs4213), 2314 HDA_CODEC_ENTRY(0x10138409, "CS8409", patch_cs8409), 1238 2315 {0} /* terminator */ 1239 2316 }; -
TabularUnified GPL/branches/uniaud32-next/alsa-kernel/pci/hda/patch_hdmi.c ¶
r693 r694 1948 1948 SND_PCI_QUIRK(0x103c, 0x870f, "HP", 1), 1949 1949 SND_PCI_QUIRK(0x103c, 0x871a, "HP", 1), 1950 SND_PCI_QUIRK(0x1462, 0xec94, "MS-7C94", 1), 1950 1951 {} 1951 1952 }; -
TabularUnified GPL/branches/uniaud32-next/alsa-kernel/pci/hda/patch_realtek.c ¶
r693 r694 299 299 { 300 300 #ifdef CONFIG_SND_HDA_GENERIC_LEDS 301 if (action == HDA_FIXUP_ACT_PR OBE)301 if (action == HDA_FIXUP_ACT_PRE_PROBE) 302 302 snd_hda_gen_add_micmute_led_cdev(codec, NULL); 303 303 #endif … … 3776 3776 report |= SND_JACK_BTN_3; 3777 3777 3778 jack->jack->button_state = report;3778 snd_hda_jack_set_button_state(codec, jack->nid, report); 3779 3779 } 3780 3780 … … 3839 3839 { 3840 3840 struct alc_spec *spec = codec->spec; 3841 hda_nid_t hp_pin; 3841 3842 3842 3843 switch (action) { … … 3845 3846 snd_hda_jack_detect_enable_callback(codec, 0x55, 3846 3847 alc_headset_btn_callback); 3847 snd_hda_jack_add_kctl(codec, 0x55, "Headset Jack", false, 3848 SND_JACK_HEADSET, alc_headset_btn_keymap); 3849 break; 3850 case HDA_FIXUP_ACT_INIT: 3848 break; 3849 case HDA_FIXUP_ACT_BUILD: 3850 hp_pin = alc_get_hp_pin(spec); 3851 if (!hp_pin || snd_hda_jack_bind_keymap(codec, 0x55, 3852 alc_headset_btn_keymap, 3853 hp_pin)) 3854 snd_hda_jack_add_kctl(codec, 0x55, "Headset Jack", 3855 false, SND_JACK_HEADSET, 3856 alc_headset_btn_keymap); 3857 3851 3858 alc_enable_headset_jack_key(codec); 3852 3859 break; … … 7275 7282 ALC256_FIXUP_ACER_HEADSET_MIC, 7276 7283 ALC285_FIXUP_IDEAPAD_S740_COEF, 7284 ALC285_FIXUP_HP_LIMIT_INT_MIC_BOOST, 7277 7285 ALC295_FIXUP_ASUS_DACS, 7278 7286 ALC295_FIXUP_HP_OMEN, … … 9116 9124 .chained = true, 9117 9125 .chain_id = ALC269_FIXUP_THINKPAD_ACPI, 9126 }, 9127 [ALC285_FIXUP_HP_LIMIT_INT_MIC_BOOST] = { 9128 .type = HDA_FIXUP_FUNC, 9129 .v.func = alc269_fixup_limit_int_mic_boost, 9130 .chained = true, 9131 .chain_id = ALC285_FIXUP_HP_MUTE_LED, 9118 9132 }, 9119 9133 [ALC295_FIXUP_ASUS_DACS] = { … … 9202 9216 SND_PCI_QUIRK(0x1025, 0x1291, "Acer Veriton Z4660G", ALC286_FIXUP_ACER_AIO_HEADSET_MIC), 9203 9217 SND_PCI_QUIRK(0x1025, 0x129c, "Acer SWIFT SF314-55", ALC256_FIXUP_ACER_HEADSET_MIC), 9218 SND_PCI_QUIRK(0x1025, 0x1300, "Acer SWIFT SF314-56", ALC256_FIXUP_ACER_MIC_NO_PRESENCE), 9204 9219 SND_PCI_QUIRK(0x1025, 0x1308, "Acer Aspire Z24-890", ALC286_FIXUP_ACER_AIO_HEADSET_MIC), 9205 9220 SND_PCI_QUIRK(0x1025, 0x132a, "Acer TravelMate B114-21", ALC233_FIXUP_ACER_HEADSET_MIC), 9206 9221 SND_PCI_QUIRK(0x1025, 0x1330, "Acer TravelMate X514-51T", ALC255_FIXUP_ACER_HEADSET_MIC), 9222 SND_PCI_QUIRK(0x1025, 0x142b, "Acer Swift SF314-42", ALC255_FIXUP_ACER_MIC_NO_PRESENCE), 9207 9223 SND_PCI_QUIRK(0x1025, 0x1430, "Acer TravelMate B311R-31", ALC256_FIXUP_ACER_MIC_NO_PRESENCE), 9208 9224 SND_PCI_QUIRK(0x1025, 0x1466, "Acer Aspire A515-56", ALC255_FIXUP_ACER_HEADPHONE_AND_MIC), … … 9369 9385 SND_PCI_QUIRK(0x103c, 0x888d, "HP ZBook Power 15.6 inch G8 Mobile Workstation PC", ALC236_FIXUP_HP_GPIO_LED), 9370 9386 SND_PCI_QUIRK(0x103c, 0x8896, "HP EliteBook 855 G8 Notebook PC", ALC285_FIXUP_HP_MUTE_LED), 9387 SND_PCI_QUIRK(0x103c, 0x8898, "HP EliteBook 845 G8 Notebook PC", ALC285_FIXUP_HP_LIMIT_INT_MIC_BOOST), 9388 SND_PCI_QUIRK(0x103c, 0x88d0, "HP Pavilion 15-eh1xxx (mainboard 88D0)", ALC287_FIXUP_HP_GPIO_LED), 9371 9389 SND_PCI_QUIRK(0x1043, 0x103e, "ASUS X540SA", ALC256_FIXUP_ASUS_MIC), 9372 9390 SND_PCI_QUIRK(0x1043, 0x103f, "ASUS TX300", ALC282_FIXUP_ASUS_TX300), … … 9552 9570 SND_PCI_QUIRK(0x17aa, 0x3176, "ThinkCentre Station", ALC283_FIXUP_HEADSET_MIC), 9553 9571 SND_PCI_QUIRK(0x17aa, 0x3178, "ThinkCentre Station", ALC283_FIXUP_HEADSET_MIC), 9572 SND_PCI_QUIRK(0x17aa, 0x31af, "ThinkCentre Station", ALC623_FIXUP_LENOVO_THINKSTATION_P340), 9554 9573 SND_PCI_QUIRK(0x17aa, 0x3818, "Lenovo C940", ALC298_FIXUP_LENOVO_SPK_VOLUME), 9555 9574 SND_PCI_QUIRK(0x17aa, 0x3827, "Ideapad S740", ALC285_FIXUP_IDEAPAD_S740_COEF), -
TabularUnified GPL/branches/uniaud32-next/alsa-kernel/pci/hda/patch_sigmatel.c ¶
r638 r694 4379 4379 spec->gen.automute_hook = stac_update_outputs; 4380 4380 4381 #ifndef TARGET_OS2 4382 if (spec->gpio_led) 4383 snd_hda_gen_add_mute_led_cdev(codec, stac_vmaster_hook); 4384 #endif 4381 4385 err = snd_hda_gen_parse_auto_config(codec, &spec->gen.autocfg); 4382 4386 if (err < 0) … … 4420 4424 #endif 4421 4425 4422 #ifdef CONFIG_SND_HDA_GENERIC_LEDS4423 if (spec->gpio_led)4424 snd_hda_gen_add_mute_led_cdev(codec, stac_vmaster_hook);4425 #endif4426 4426 if (spec->aloopback_ctl && 4427 4427 snd_hda_get_bool_hint(codec, "loopback") == 1) { -
TabularUnified GPL/branches/uniaud32-next/alsa-kernel/pci/hda/thinkpad_helper.c ¶
r629 r694 19 19 const struct hda_fixup *fix, int action) 20 20 { 21 if (action == HDA_FIXUP_ACT_PR OBE) {21 if (action == HDA_FIXUP_ACT_PRE_PROBE) { 22 22 if (!is_thinkpad(codec)) 23 23 return; -
TabularUnified GPL/branches/uniaud32-next/include/linux/kmod.h ¶
r32 r694 3 3 #ifndef _LINUX_KMOD_H 4 4 #define _LINUX_KMOD_H 5 #include <asm/errno.h> 6 static inline int request_module(const char *name, ...) { return -ENOSYS; } 5 7 6 8 #endif /* _LINUX_KMOD_H */ -
TabularUnified GPL/branches/uniaud32-next/include/linux/moduleparam.h ¶
r647 r694 62 62 ulong, charp, bool or invbool, or XXX if you define param_get_XXX, 63 63 param_set_XXX and param_check_XXX. */ 64 #define module_param_named(name, value, type, perm) \ 65 param_check_##type(name, &(value)); \ 66 module_param_call(name, param_set_##type, param_get_##type, &value, perm) 64 #define module_param_named(name, value, type, perm) 67 65 68 66 #ifndef TARGET_OS2 -
TabularUnified GPL/branches/uniaud32-next/include/linux/rwsem.h ¶
r647 r694 14 14 #define up_read(x) up(x) 15 15 #define up_write(x) up(x) 16 #define downgrade_write 16 17 17 18 static inline int down_write_trylock(struct rw_semaphore *sem) {return 0;} -
TabularUnified GPL/branches/uniaud32-next/include/linux/slab.h ¶
r692 r694 116 116 krealloc_array(void *p, size_t new_n, size_t new_size, gfp_t flags) 117 117 { 118 size_t bytes;118 // size_t bytes; 119 119 120 120 // if (check_mul_overflow(new_n, new_size, &bytes)) 121 121 // return NULL; 122 122 123 return krealloc(p, bytes, flags);123 return krealloc(p, new_n*new_size, flags); 124 124 } 125 125
Note:
See TracChangeset
for help on using the changeset viewer.