如何区分音量控制中:Speaker(播放控制)里的microphone和Wave in(录音控制)里的microphone?

shadowWind 2004-09-22 11:27:07
我有两个声卡A,B.
A: 只有Speaker音量控制中有microphone这一项。

我能够用mixer系列函数控制它的volume和mute,
只要把 MIXERLINE结构中的dwComponentType设为MIXERLINE_COMPONENTTYPE_SRC_MICROPHONE 就行

B: Speaker音量控制和Wave in音量控制中都有 microphone这一项

我用了与A中同样的代码,受控制的只有 wave in中的microphone
怎样才能够控制Speaker中的microphone的volume和mute?

我已经是过其他所有可选的dwComponentType,都不行。
我的想法:
根据A中的现象,dwComponentType应该还是用MIXERLINE_COMPONENTTYPE_SRC_MICROPHONE;
应该有别的参数设置可以区分这两个microphone

但我不知道还有哪个地方需要设置,请指点,多谢!


...全文
476 6 打赏 收藏 转发到动态 举报
写回复
用AI写文章
6 条回复
切换为时间正序
请发表友善的回复…
发表回复
shadowWind 2004-10-13
  • 打赏
  • 举报
回复
又研究了一遍vfwMixer的代码。
自己解决了,做个总结,顺便结贴。

原来获取ControlID的方法中:
使用::mixerGetLineInfo()是第三个参数用了MIXER_GETLINEINFOF_COMPONENTTYPE,找了很久没有发现这样有什么途径可以区分 dwComponentType=MIXERLINE_COMPONENTTYPE_SRC_MICROPHONE 是到底取得是录音设备还是播放设备。

因此,改用了vfwMixer中获取ControlID的方法。
即:::mixerGetLineInfo()是第三个参数用MIXER_GETLINEINFOF_DESTINATION
然后,MIXERLINE数据的dwDestination成员分别置0和1时,分别代表的是录音和播放设备。
这样在遍历中,当dwDestination为0时,若相关联的MixerControl的dwControlType为MIXERLINE_COMPONENTTYPE_SRC_MICROPHONE,则找到了一个mic录音源
反之,dwDestiantion为1时,。。。。。,则是播放控制中的microphone.

下面附相关代码(参照vfwMixer代码做了修改):

BOOL GetControlID(DWORD ComponentType, DWORD ControlType ,BOOL DESTINATION)
{
MMRESULT err;
unsigned long j, k, l;

/* Get info about all the destination lines for this mixer */
// for(j=0;j<m_mxcaps.cDestinations;j++){
if (DESTINATION) j=0;
else j=1;

MIXERLINE dmixerline;
dmixerline.cbStruct = sizeof(MIXERLINE);
dmixerline.dwDestination = j;
if ((err = mixerGetLineInfo((HMIXEROBJ)m_hMixer,&dmixerline, MIXER_GETLINEINFOF_DESTINATION))) {
TRACE0("Error getting destination lines\n");
}
else{
/* Get info about all the controls for this dest line */
MIXERLINECONTROLS destLineControls;
MIXERCONTROL * destControlArray = (MIXERCONTROL*)malloc(dmixerline.cControls * sizeof(MIXERCONTROL));
destLineControls.cbStruct = sizeof(MIXERLINECONTROLS);
destLineControls.cControls = dmixerline.cControls;
destLineControls.dwLineID = dmixerline.dwLineID;
destLineControls.pamxctrl = destControlArray;
destLineControls.cbmxctrl = sizeof(MIXERCONTROL);
if ((err = mixerGetLineControls((HMIXEROBJ)m_hMixer, &destLineControls, MIXER_GETLINECONTROLSF_ALL))){
TRACE0("Error getting destination line controls");
}
else{
for(l=0;l<dmixerline.cControls;l++){
if ( destControlArray[l].dwControlType==ComponentType ) {
m_strDstLineName = dmixerline.szName;
if ( MIXERCONTROL_CONTROLTYPE_VOLUME==destControlArray[l].dwControlType) {
m_dwMinimum = destControlArray[l].Bounds.dwMinimum;
m_dwMaximum = destControlArray[l].Bounds.dwMaximum;
m_dwVolumeControlID = destControlArray[l].dwControlID;
m_strVolumeControlName = destControlArray[l].szName;
return TRUE;
}
else if ( MIXERCONTROL_CONTROLTYPE_MUTE==destControlArray[l].dwControlType) {
m_strMuteControlName = destControlArray[l].szName;
m_dwMuteControlID = destControlArray[l].dwControlID;
return TRUE;
}
else if ( MIXERCONTROL_CONTROLTYPE_ONOFF==destControlArray[l].dwControlType) {
m_dwONOFFControlID = destControlArray[l].dwControlID;
return TRUE;
}
}
}
}
/* Get info about all the source lines for this dest line */
for(k=0;k<dmixerline.cConnections;k++){
MIXERLINE smixerline;
smixerline.cbStruct = sizeof(MIXERLINE);
smixerline.dwDestination = j;
smixerline.dwSource = k;
if ((err = mixerGetLineInfo((HMIXEROBJ)m_hMixer, &smixerline, MIXER_GETLINEINFOF_SOURCE))){
TRACE0("Error getting source lines\n");
}
else{
/* get info about all the controls for this source line */
MIXERLINECONTROLS sourceLineControls;
MIXERCONTROL * sourceControlArray = (MIXERCONTROL*)malloc(smixerline.cControls * sizeof(MIXERCONTROL));
sourceLineControls.cbStruct = sizeof(MIXERLINECONTROLS);
sourceLineControls.cControls = smixerline.cControls;
sourceLineControls.dwLineID = smixerline.dwLineID;
sourceLineControls.pamxctrl = sourceControlArray;
sourceLineControls.cbmxctrl = sizeof(MIXERCONTROL);
if ((err = mixerGetLineControls((HMIXEROBJ)m_hMixer, &sourceLineControls, MIXER_GETLINECONTROLSF_ALL))){
TRACE0("Error getting source line controls");
}
else{
for(l=0;l<smixerline.cControls;l++){
if ( smixerline.dwComponentType==ComponentType && sourceControlArray[l].dwControlType==ControlType) {
m_strDstLineName = smixerline.szName;
if ( MIXERCONTROL_CONTROLTYPE_VOLUME==sourceControlArray[l].dwControlType) {
m_dwMinimum = sourceControlArray[l].Bounds.dwMinimum;
m_dwMaximum = sourceControlArray[l].Bounds.dwMaximum;
m_dwVolumeControlID = sourceControlArray[l].dwControlID;
m_strVolumeControlName = sourceControlArray[l].szName;
return TRUE;
}
else if( MIXERCONTROL_CONTROLTYPE_MUTE==sourceControlArray[l].dwControlType ){
m_strMuteControlName = sourceControlArray[l].szName;
m_dwMuteControlID = sourceControlArray[l].dwControlID;
return TRUE;
}
else if ( MIXERCONTROL_CONTROLTYPE_ONOFF==sourceControlArray[l].dwControlType) {
m_dwONOFFControlID = sourceControlArray[l].dwControlID;
return TRUE;
}
}
}
}
}
}
}
// }
return FALSE;
}


shadowWind 2004-10-10
  • 打赏
  • 举报
回复
up
:(
pepsi1980 2004-10-10
  • 打赏
  • 举报
回复
看完题目头就晕了.......
shadowWind 2004-09-24
  • 打赏
  • 举报
回复
up
nwpulipeng 2004-09-24
  • 打赏
  • 举报
回复
帮顶混分
DentistryDoctor 2004-09-22
  • 打赏
  • 举报
回复
mixerGetID?来同的设置ID不同。
课程简述  Android是目前最为流行的移动操作系统之一,它的开发涉及到多个知识领域。本课程将深入介绍Android系统启动过程的重要组成部分——init进程,并探讨与之相关的多项关键技术。我们还将提供实际的开发案例,以RK3399开发板为例,通过演示Android产品配置、init启动流程、selinux权限管理、init.rc启动脚本定制等实际案例,让学员深入理解这些技术在实际产品开发的应用场景和实现方法,提高学员的实际开发能力和经验,从而更好地应对实际产品开发遇到的问题和挑战。知识运用方向学习Android启动方面的知识,可以参与如下实际开发工作项:启动流程定制: 根据产品需求调整Android启动流程,包括修改init.rc脚本、修改启动顺序和等待时间、加入自定义服务等。属性系统定制: 通过Android属性系统定制化启动流程,例如增加产品版本信息、定制开机音量等。日志系统分析: 掌握日志的捕捉、分析和排错技术,在启动过程,需要加入调试信息来方便开发人员进行调试,同时需要进行日志的优化,避免日志输出过多占用过多的系统资源。selinux安全策略定制:在Android系统,selinux是一种安全机制,用于保护系统的敏感资源和数据。在实际开发,可能需要对selinux策略进行定制,以确保系统的安全性和稳定性。课程内容主要内容简述1, RK3399 开发板操作这部分内容重点介绍如何在FIreFly开发板上将Android 10系统运行起来, 包含编译FireFly的Android源码下载和编译, 镜像烧录运行,内核和模块编译,以及RK3399内核启动init进程的过程。2, 产品定制这部分讲解获取到方案商或者原厂提供的源码后, 如何定制一个新的产品,产品配置文件和模型, 原始代码的配置文件和定制化东西3, Android日志代码编写之前讲过Android的日志系统, 并没涉及到代码编写, 这个部分重点讲解C/C++, java代码编写日志的API和代码4, 属性系统在Android,属性使用的非常频繁的,可以用来作为进程间通信,也可以用于一些行为控制, 这个部分会重点介绍属性系统框架, API接口, 属性文件等知识点5,selinux进程对文件进行访问时,Android 4.3就开始集成了selinux权限管控, 如果需要启动某个脚本或者服务, selinux的配置就避免不了,并且Android8之后, Android系统对进程访问的权限管控的非常严格。6, init.rc脚本Android定义的一种脚本, 改脚本是有init进程启动, 是非常重要的一个脚本, 会包含系统的其他很多脚本, 在我们系统开发时, 我们经常通过这个脚本进行一些定制化动作。7, init进程代码分析想要了解一个系统,就必须对源码进行分析和理解, 这个章节,带大家去跟读init进程代码, 这样,换了另外一个Android版本,完全就可以去读代码, 知道有什么变化。 

2,641

社区成员

发帖
与我相关
我的任务
社区描述
VC/MFC 硬件/系统
社区管理员
  • 硬件/系统社区
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧