pulseaudio asyncdevicelist
Linux/Linux 일반 :
2019. 11. 29. 15:58
반응형
//gcc ./asyncdevicelist.c -lpulse
#include <stdio.h>
#include <string.h>
#include <pulse/pulseaudio.h>
#define MAX_DEVICE_NUM 16
// Field list is here: http://0pointer.de/lennart/projects/pulseaudio/doxygen/structpa__sink__info.html
typedef struct pa_devicelist {
uint8_t initialized;
char name[512];
uint32_t index;
char description[256];
} pa_devicelist_t;
void pa_state_cb(pa_context *c, void *userdata);
void pa_sinklist_cb(pa_context *c, const pa_sink_info *l, int eol, void *userdata);
void pa_sourcelist_cb(pa_context *c, const pa_source_info *l, int eol, void *userdata);
void pa_cardlist_cb(pa_context *c, const pa_card_info *l, int eol, void *userdata);
int pa_get_devicelist(pa_devicelist_t *input, pa_devicelist_t *output, pa_devicelist_t *card);
int main(int argc, char *argv[]) {
int ctr;
// This is where we'll store the input device list
pa_devicelist_t pa_input_devicelist[MAX_DEVICE_NUM];
// This is where we'll store the output device list
pa_devicelist_t pa_output_devicelist[MAX_DEVICE_NUM];
pa_devicelist_t pa_card_devicelist[MAX_DEVICE_NUM];
if (pa_get_devicelist(pa_input_devicelist, pa_output_devicelist, pa_card_devicelist) < 0) {
fprintf(stderr, "failed to get device list\n");
return 1;
}
for (ctr = 0; ctr < MAX_DEVICE_NUM; ctr++) {
if (! pa_output_devicelist[ctr].initialized) {
break;
}
printf("=======[ Output Device #%d ]=======\n", ctr+1);
printf("Description: %s\n", pa_output_devicelist[ctr].description);
printf("Name: %s\n", pa_output_devicelist[ctr].name);
printf("Index: %d\n", pa_output_devicelist[ctr].index);
printf("\n");
}
for (ctr = 0; ctr < MAX_DEVICE_NUM; ctr++) {
if (! pa_input_devicelist[ctr].initialized) {
break;
}
printf("=======[ Input Device #%d ]=======\n", ctr+1);
printf("Description: %s\n", pa_input_devicelist[ctr].description);
printf("Name: %s\n", pa_input_devicelist[ctr].name);
printf("Index: %d\n", pa_input_devicelist[ctr].index);
printf("\n");
}
for (ctr = 0; ctr < MAX_DEVICE_NUM; ctr++) {
if (! pa_input_devicelist[ctr].initialized) {
break;
}
printf("=======[ Card Device #%d ]=======\n", ctr+1);
printf("Driver Name: %s\n", pa_card_devicelist[ctr].description);
printf("Name: %s\n", pa_card_devicelist[ctr].name);
printf("Index: %d\n", pa_card_devicelist[ctr].index);
printf("\n");
}
return 0;
}
int pa_get_devicelist(pa_devicelist_t *input, pa_devicelist_t *output, pa_devicelist_t *card) {
//int pa_get_devicelist(pa_devicelist_t *input, pa_devicelist_t *output) {
// Define our pulse audio loop and connection variables
pa_mainloop *pa_ml;
pa_mainloop_api *pa_mlapi;
pa_operation *pa_op;
pa_context *pa_ctx;
// We'll need these state variables to keep track of our requests
int state = 0;
int pa_ready = 0;
// Initialize our device lists
memset(input, 0, sizeof(pa_devicelist_t) * MAX_DEVICE_NUM);
memset(output, 0, sizeof(pa_devicelist_t) * MAX_DEVICE_NUM);
memset(card, 0, sizeof(pa_devicelist_t) * MAX_DEVICE_NUM);
// Create a mainloop API and connection to the default server
pa_ml = pa_mainloop_new();
pa_mlapi = pa_mainloop_get_api(pa_ml);
pa_ctx = pa_context_new(pa_mlapi, "test");
// This function connects to the pulse server
pa_context_connect(pa_ctx, NULL, 0, NULL);
// This function defines a callback so the server will tell us it's state.
// Our callback will wait for the state to be ready. The callback will
// modify the variable to 1 so we know when we have a connection and it's
// ready.
// If there's an error, the callback will set pa_ready to 2
pa_context_set_state_callback(pa_ctx, pa_state_cb, &pa_ready);
printf("====end state cb\n");
// Now we'll enter into an infinite loop until we get the data we receive
// or if there's an error
for (;;) {
// We can't do anything until PA is ready, so just iterate the mainloop
// and continue
if (pa_ready == 0) {
pa_mainloop_iterate(pa_ml, 1, NULL);
continue;
}
// We couldn't get a connection to the server, so exit out
if (pa_ready == 2) {
printf("mainloop out\n");
pa_context_disconnect(pa_ctx);
pa_context_unref(pa_ctx);
pa_mainloop_free(pa_ml);
return -1;
}
// At this point, we're connected to the server and ready to make
// requests
switch (state) {
// State 0: we haven't done anything yet
case 0:
// This sends an operation to the server. pa_sinklist_info is
// our callback function and a pointer to our devicelist will
// be passed to the callback The operation ID is stored in the
// pa_op variable
pa_op = pa_context_get_sink_info_list(pa_ctx,
pa_sinklist_cb,
output
);
// Update state for next iteration through the loop
printf("go to next state %d\n",state);
state++;
break;
case 1:
// Now we wait for our operation to complete. When it's
// complete our pa_output_devicelist is filled out, and we move
// along to the next state
if (pa_operation_get_state(pa_op) == PA_OPERATION_DONE) {
pa_operation_unref(pa_op);
// Now we perform another operation to get the source
// (input device) list just like before. This time we pass
// a pointer to our input structure
pa_op = pa_context_get_source_info_list(pa_ctx,
pa_sourcelist_cb,
input
);
// Update the state so we know what to do next
printf("go to next state %d\n",state);
state++;
}
break;
case 2:
// Now we wait for our operation to complete. When it's
// complete our pa_output_devicelist is filled out, and we move
// along to the next state
if (pa_operation_get_state(pa_op) == PA_OPERATION_DONE) {
pa_operation_unref(pa_op);
// Now we perform another operation to get the source
// (input device) list just like before. This time we pass
// a pointer to our input structure
pa_op = pa_context_get_card_info_list(pa_ctx,
pa_cardlist_cb,
card
);
// Update the state so we know what to do next
printf("go to next state %d\n",state);
state++;
}
break;
case 3:
if (pa_operation_get_state(pa_op) == PA_OPERATION_DONE) {
// Now we're done, clean up and disconnect and return
printf("### mainloop disconnect 1\n");
pa_operation_unref(pa_op);
printf("### mainloop disconnect 2\n");
pa_context_disconnect(pa_ctx);
printf("### mainloop disconnect 3\n");
pa_context_unref(pa_ctx);
printf("### mainloop disconnect 4\n");
pa_mainloop_free(pa_ml);
return 0;
}
break;
default:
// We should never see this state
fprintf(stderr, "in state %d\n", state);
return -1;
}
// Iterate the main loop and go again. The second argument is whether
// or not the iteration should block until something is ready to be
// done. Set it to zero for non-blocking.
pa_mainloop_iterate(pa_ml, 1, NULL);
}
}
// This callback gets called when our context changes state. We really only
// care about when it's ready or if it has failed
void pa_state_cb(pa_context *c, void *userdata) {
pa_context_state_t state;
int *pa_ready = userdata;
state = pa_context_get_state(c);
fprintf(stderr, "state:%d, pa_ready:%d\n", state, *pa_ready);
switch (state) {
// There are just here for reference
case PA_CONTEXT_UNCONNECTED:
fprintf(stderr, "PA_CONTEXT_UNCONNECTED\n");
break;
case PA_CONTEXT_CONNECTING:
fprintf(stderr, "PA_CONTEXT_CONNECTING\n");
break;
case PA_CONTEXT_AUTHORIZING:
fprintf(stderr, "PA_CONTEXT_AUTHORIZING\n");
break;
case PA_CONTEXT_SETTING_NAME:
fprintf(stderr, "PA_CONTEXT_SETTING_NAME\n");
break;
default:
break;
case PA_CONTEXT_FAILED:
fprintf(stderr, "PA_CONTEXT_FAILED\n");
break;
case PA_CONTEXT_TERMINATED:
fprintf(stderr, "PA_CONTEXT_TERMINATED\n");
*pa_ready = 2;
break;
case PA_CONTEXT_READY:
fprintf(stderr, "PA_CONTEXT_READY\n");
*pa_ready = 1;
break;
}
}
// pa_mainloop will call this function when it's ready to tell us about a sink.
// Since we're not threading, there's no need for mutexes on the devicelist
// structure
void pa_sinklist_cb(pa_context *c, const pa_sink_info *l, int eol, void *userdata) {
printf("*** sink cb\n");
pa_devicelist_t *pa_devicelist = userdata;
int ctr = 0;
// If eol is set to a positive number, you're at the end of the list
if (eol > 0) {
return;
}
// We know we've allocated 16 slots to hold devices. Loop through our
// structure and find the first one that's "uninitialized." Copy the
// contents into it and we're done. If we receive more than 16 devices,
// they're going to get dropped. You could make this dynamically allocate
// space for the device list, but this is a simple example.
for (ctr = 0; ctr < MAX_DEVICE_NUM; ctr++) {
if (! pa_devicelist[ctr].initialized) {
strncpy(pa_devicelist[ctr].name, l->name, 511);
strncpy(pa_devicelist[ctr].description, l->description, 255);
pa_devicelist[ctr].index = l->index;
pa_devicelist[ctr].initialized = 1;
break;
}
}
}
// See above. This callback is pretty much identical to the previous
void pa_sourcelist_cb(pa_context *c, const pa_source_info *l, int eol, void *userdata) {
printf("+++ src cb\n");
pa_devicelist_t *pa_devicelist = userdata;
int ctr = 0;
if (eol > 0) {
return;
}
for (ctr = 0; ctr < MAX_DEVICE_NUM; ctr++) {
if (! pa_devicelist[ctr].initialized) {
strncpy(pa_devicelist[ctr].name, l->name, 511);
strncpy(pa_devicelist[ctr].description, l->description, 255);
pa_devicelist[ctr].index = l->index;
pa_devicelist[ctr].initialized = 1;
break;
}
}
}
void pa_cardlist_cb(pa_context *c, const pa_card_info *l, int eol, void *userdata){
printf("+++ card cb\n");
pa_devicelist_t *pa_devicelist = userdata;
int ctr = 0;
if (eol > 0) {
return;
}
for (ctr = 0; ctr < MAX_DEVICE_NUM; ctr++) {
if (! pa_devicelist[ctr].initialized) {
strncpy(pa_devicelist[ctr].name, l->name, 511);
strncpy(pa_devicelist[ctr].description, l->driver, 255);
pa_devicelist[ctr].index = l->index;
pa_devicelist[ctr].initialized = 1;
break;
}
}
}
반응형
'Linux > Linux 일반' 카테고리의 다른 글
ubuntu mkimage, adb, fastboot 설치 (0) | 2020.02.27 |
---|---|
PulseAudio build (0) | 2019.11.26 |
pulseaudio device list (0) | 2019.09.17 |