Blame ext/pulse/pulseutil.c

Packit 1f69a5
/*
Packit 1f69a5
 *  GStreamer pulseaudio plugin
Packit 1f69a5
 *
Packit 1f69a5
 *  Copyright (c) 2004-2008 Lennart Poettering
Packit 1f69a5
 *
Packit 1f69a5
 *  gst-pulse is free software; you can redistribute it and/or modify
Packit 1f69a5
 *  it under the terms of the GNU Lesser General Public License as
Packit 1f69a5
 *  published by the Free Software Foundation; either version 2.1 of the
Packit 1f69a5
 *  License, or (at your option) any later version.
Packit 1f69a5
 *
Packit 1f69a5
 *  gst-pulse is distributed in the hope that it will be useful, but
Packit 1f69a5
 *  WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 1f69a5
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Packit 1f69a5
 *  Lesser General Public License for more details.
Packit 1f69a5
 *
Packit 1f69a5
 *  You should have received a copy of the GNU Lesser General Public
Packit 1f69a5
 *  License along with gst-pulse; if not, write to the Free Software
Packit 1f69a5
 *  Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301
Packit 1f69a5
 *  USA.
Packit 1f69a5
 */
Packit 1f69a5
Packit 1f69a5
#ifdef HAVE_CONFIG_H
Packit 1f69a5
#include "config.h"
Packit 1f69a5
#endif
Packit 1f69a5
Packit 1f69a5
#include <gst/audio/audio.h>
Packit 1f69a5
Packit 1f69a5
#include "pulseutil.h"
Packit 1f69a5
Packit 1f69a5
#ifdef HAVE_UNISTD_H
Packit 1f69a5
# include <unistd.h>            /* getpid on UNIX */
Packit 1f69a5
#endif
Packit 1f69a5
#ifdef HAVE_PROCESS_H
Packit 1f69a5
# include <process.h>           /* getpid on win32 */
Packit 1f69a5
#endif
Packit 1f69a5
Packit 1f69a5
static const struct
Packit 1f69a5
{
Packit 1f69a5
  GstAudioChannelPosition gst_pos;
Packit 1f69a5
  pa_channel_position_t pa_pos;
Packit 1f69a5
} gst_pa_pos_table[] = {
Packit 1f69a5
  {
Packit 1f69a5
  GST_AUDIO_CHANNEL_POSITION_MONO, PA_CHANNEL_POSITION_MONO}, {
Packit 1f69a5
  GST_AUDIO_CHANNEL_POSITION_FRONT_LEFT, PA_CHANNEL_POSITION_FRONT_LEFT}, {
Packit 1f69a5
  GST_AUDIO_CHANNEL_POSITION_FRONT_RIGHT, PA_CHANNEL_POSITION_FRONT_RIGHT}, {
Packit 1f69a5
  GST_AUDIO_CHANNEL_POSITION_REAR_CENTER, PA_CHANNEL_POSITION_REAR_CENTER}, {
Packit 1f69a5
  GST_AUDIO_CHANNEL_POSITION_REAR_LEFT, PA_CHANNEL_POSITION_REAR_LEFT}, {
Packit 1f69a5
  GST_AUDIO_CHANNEL_POSITION_REAR_RIGHT, PA_CHANNEL_POSITION_REAR_RIGHT}, {
Packit 1f69a5
  GST_AUDIO_CHANNEL_POSITION_LFE1, PA_CHANNEL_POSITION_LFE}, {
Packit 1f69a5
  GST_AUDIO_CHANNEL_POSITION_FRONT_CENTER, PA_CHANNEL_POSITION_FRONT_CENTER}, {
Packit 1f69a5
  GST_AUDIO_CHANNEL_POSITION_FRONT_LEFT_OF_CENTER,
Packit 1f69a5
        PA_CHANNEL_POSITION_FRONT_LEFT_OF_CENTER}, {
Packit 1f69a5
  GST_AUDIO_CHANNEL_POSITION_FRONT_RIGHT_OF_CENTER,
Packit 1f69a5
        PA_CHANNEL_POSITION_FRONT_RIGHT_OF_CENTER}, {
Packit 1f69a5
  GST_AUDIO_CHANNEL_POSITION_SIDE_LEFT, PA_CHANNEL_POSITION_SIDE_LEFT}, {
Packit 1f69a5
  GST_AUDIO_CHANNEL_POSITION_SIDE_RIGHT, PA_CHANNEL_POSITION_SIDE_RIGHT}, {
Packit 1f69a5
  GST_AUDIO_CHANNEL_POSITION_TOP_CENTER, PA_CHANNEL_POSITION_TOP_CENTER}, {
Packit 1f69a5
  GST_AUDIO_CHANNEL_POSITION_TOP_FRONT_LEFT,
Packit 1f69a5
        PA_CHANNEL_POSITION_TOP_FRONT_LEFT}, {
Packit 1f69a5
  GST_AUDIO_CHANNEL_POSITION_TOP_FRONT_RIGHT,
Packit 1f69a5
        PA_CHANNEL_POSITION_TOP_FRONT_RIGHT}, {
Packit 1f69a5
  GST_AUDIO_CHANNEL_POSITION_TOP_FRONT_CENTER,
Packit 1f69a5
        PA_CHANNEL_POSITION_TOP_FRONT_CENTER}, {
Packit 1f69a5
  GST_AUDIO_CHANNEL_POSITION_TOP_REAR_LEFT, PA_CHANNEL_POSITION_TOP_REAR_LEFT}, {
Packit 1f69a5
  GST_AUDIO_CHANNEL_POSITION_TOP_REAR_RIGHT,
Packit 1f69a5
        PA_CHANNEL_POSITION_TOP_REAR_RIGHT}, {
Packit 1f69a5
  GST_AUDIO_CHANNEL_POSITION_TOP_REAR_CENTER,
Packit 1f69a5
        PA_CHANNEL_POSITION_TOP_REAR_CENTER}, {
Packit 1f69a5
  GST_AUDIO_CHANNEL_POSITION_NONE, PA_CHANNEL_POSITION_INVALID}
Packit 1f69a5
};
Packit 1f69a5
Packit 1f69a5
static gboolean
Packit 1f69a5
gstaudioformat_to_pasampleformat (GstAudioFormat format,
Packit 1f69a5
    pa_sample_format_t * sf)
Packit 1f69a5
{
Packit 1f69a5
  switch (format) {
Packit 1f69a5
    case GST_AUDIO_FORMAT_U8:
Packit 1f69a5
      *sf = PA_SAMPLE_U8;
Packit 1f69a5
      break;
Packit 1f69a5
    case GST_AUDIO_FORMAT_S16LE:
Packit 1f69a5
      *sf = PA_SAMPLE_S16LE;
Packit 1f69a5
      break;
Packit 1f69a5
    case GST_AUDIO_FORMAT_S16BE:
Packit 1f69a5
      *sf = PA_SAMPLE_S16BE;
Packit 1f69a5
      break;
Packit 1f69a5
    case GST_AUDIO_FORMAT_F32LE:
Packit 1f69a5
      *sf = PA_SAMPLE_FLOAT32LE;
Packit 1f69a5
      break;
Packit 1f69a5
    case GST_AUDIO_FORMAT_F32BE:
Packit 1f69a5
      *sf = PA_SAMPLE_FLOAT32BE;
Packit 1f69a5
      break;
Packit 1f69a5
    case GST_AUDIO_FORMAT_S32LE:
Packit 1f69a5
      *sf = PA_SAMPLE_S32LE;
Packit 1f69a5
      break;
Packit 1f69a5
    case GST_AUDIO_FORMAT_S32BE:
Packit 1f69a5
      *sf = PA_SAMPLE_S32BE;
Packit 1f69a5
      break;
Packit 1f69a5
    case GST_AUDIO_FORMAT_S24LE:
Packit 1f69a5
      *sf = PA_SAMPLE_S24LE;
Packit 1f69a5
      break;
Packit 1f69a5
    case GST_AUDIO_FORMAT_S24BE:
Packit 1f69a5
      *sf = PA_SAMPLE_S24BE;
Packit 1f69a5
      break;
Packit 1f69a5
    case GST_AUDIO_FORMAT_S24_32LE:
Packit 1f69a5
      *sf = PA_SAMPLE_S24_32LE;
Packit 1f69a5
      break;
Packit 1f69a5
    case GST_AUDIO_FORMAT_S24_32BE:
Packit 1f69a5
      *sf = PA_SAMPLE_S24_32BE;
Packit 1f69a5
      break;
Packit 1f69a5
    default:
Packit 1f69a5
      return FALSE;
Packit 1f69a5
  }
Packit 1f69a5
  return TRUE;
Packit 1f69a5
}
Packit 1f69a5
Packit 1f69a5
gboolean
Packit 1f69a5
gst_pulse_fill_sample_spec (GstAudioRingBufferSpec * spec, pa_sample_spec * ss)
Packit 1f69a5
{
Packit 1f69a5
  if (spec->type == GST_AUDIO_RING_BUFFER_FORMAT_TYPE_RAW) {
Packit 1f69a5
    if (!gstaudioformat_to_pasampleformat (GST_AUDIO_INFO_FORMAT (&spec->info),
Packit 1f69a5
            &ss->format))
Packit 1f69a5
      return FALSE;
Packit 1f69a5
  } else if (spec->type == GST_AUDIO_RING_BUFFER_FORMAT_TYPE_MU_LAW) {
Packit 1f69a5
    ss->format = PA_SAMPLE_ULAW;
Packit 1f69a5
  } else if (spec->type == GST_AUDIO_RING_BUFFER_FORMAT_TYPE_A_LAW) {
Packit 1f69a5
    ss->format = PA_SAMPLE_ALAW;
Packit 1f69a5
  } else
Packit 1f69a5
    return FALSE;
Packit 1f69a5
Packit 1f69a5
  ss->channels = GST_AUDIO_INFO_CHANNELS (&spec->info);
Packit 1f69a5
  ss->rate = GST_AUDIO_INFO_RATE (&spec->info);
Packit 1f69a5
Packit 1f69a5
  if (!pa_sample_spec_valid (ss))
Packit 1f69a5
    return FALSE;
Packit 1f69a5
Packit 1f69a5
  return TRUE;
Packit 1f69a5
}
Packit 1f69a5
Packit 1f69a5
gboolean
Packit 1f69a5
gst_pulse_fill_format_info (GstAudioRingBufferSpec * spec, pa_format_info ** f,
Packit 1f69a5
    guint * channels)
Packit 1f69a5
{
Packit 1f69a5
  pa_format_info *format;
Packit 1f69a5
  pa_sample_format_t sf = PA_SAMPLE_INVALID;
Packit 1f69a5
  GstAudioInfo *ainfo = &spec->info;
Packit 1f69a5
Packit 1f69a5
  format = pa_format_info_new ();
Packit 1f69a5
Packit 1f69a5
  if (spec->type == GST_AUDIO_RING_BUFFER_FORMAT_TYPE_MU_LAW) {
Packit 1f69a5
    format->encoding = PA_ENCODING_PCM;
Packit 1f69a5
    sf = PA_SAMPLE_ULAW;
Packit 1f69a5
  } else if (spec->type == GST_AUDIO_RING_BUFFER_FORMAT_TYPE_A_LAW) {
Packit 1f69a5
    format->encoding = PA_ENCODING_PCM;
Packit 1f69a5
    sf = PA_SAMPLE_ALAW;
Packit 1f69a5
  } else if (spec->type == GST_AUDIO_RING_BUFFER_FORMAT_TYPE_RAW) {
Packit 1f69a5
    format->encoding = PA_ENCODING_PCM;
Packit 1f69a5
    if (!gstaudioformat_to_pasampleformat (GST_AUDIO_INFO_FORMAT (ainfo), &sf))
Packit 1f69a5
      goto fail;
Packit 1f69a5
  } else if (spec->type == GST_AUDIO_RING_BUFFER_FORMAT_TYPE_AC3) {
Packit 1f69a5
    format->encoding = PA_ENCODING_AC3_IEC61937;
Packit 1f69a5
  } else if (spec->type == GST_AUDIO_RING_BUFFER_FORMAT_TYPE_EAC3) {
Packit 1f69a5
    format->encoding = PA_ENCODING_EAC3_IEC61937;
Packit 1f69a5
  } else if (spec->type == GST_AUDIO_RING_BUFFER_FORMAT_TYPE_DTS) {
Packit 1f69a5
    format->encoding = PA_ENCODING_DTS_IEC61937;
Packit 1f69a5
  } else if (spec->type == GST_AUDIO_RING_BUFFER_FORMAT_TYPE_MPEG) {
Packit 1f69a5
    format->encoding = PA_ENCODING_MPEG_IEC61937;
Packit 1f69a5
#if PA_CHECK_VERSION(3,99,0)
Packit 1f69a5
  } else if (spec->type == GST_AUDIO_RING_BUFFER_FORMAT_TYPE_MPEG2_AAC) {
Packit 1f69a5
    format->encoding = PA_ENCODING_MPEG2_AAC_IEC61937;
Packit 1f69a5
  } else if (spec->type == GST_AUDIO_RING_BUFFER_FORMAT_TYPE_MPEG4_AAC) {
Packit 1f69a5
    /* HACK. treat MPEG4 AAC as MPEG2 AAC for the moment */
Packit 1f69a5
    format->encoding = PA_ENCODING_MPEG2_AAC_IEC61937;
Packit 1f69a5
#endif
Packit 1f69a5
  } else {
Packit 1f69a5
    goto fail;
Packit 1f69a5
  }
Packit 1f69a5
Packit 1f69a5
  if (format->encoding == PA_ENCODING_PCM) {
Packit 1f69a5
    pa_format_info_set_sample_format (format, sf);
Packit 1f69a5
    pa_format_info_set_channels (format, GST_AUDIO_INFO_CHANNELS (ainfo));
Packit 1f69a5
  }
Packit 1f69a5
Packit 1f69a5
  pa_format_info_set_rate (format, GST_AUDIO_INFO_RATE (ainfo));
Packit 1f69a5
Packit 1f69a5
  if (!pa_format_info_valid (format))
Packit 1f69a5
    goto fail;
Packit 1f69a5
Packit 1f69a5
  *f = format;
Packit 1f69a5
  *channels = GST_AUDIO_INFO_CHANNELS (ainfo);
Packit 1f69a5
Packit 1f69a5
  return TRUE;
Packit 1f69a5
Packit 1f69a5
fail:
Packit 1f69a5
  if (format)
Packit 1f69a5
    pa_format_info_free (format);
Packit 1f69a5
  return FALSE;
Packit 1f69a5
}
Packit 1f69a5
Packit 1f69a5
const char *
Packit 1f69a5
gst_pulse_sample_format_to_caps_format (pa_sample_format_t sf)
Packit 1f69a5
{
Packit 1f69a5
  switch (sf) {
Packit 1f69a5
    case PA_SAMPLE_U8:
Packit 1f69a5
      return "U8";
Packit 1f69a5
Packit 1f69a5
    case PA_SAMPLE_S16LE:
Packit 1f69a5
      return "S16LE";
Packit 1f69a5
Packit 1f69a5
    case PA_SAMPLE_S16BE:
Packit 1f69a5
      return "S16BE";
Packit 1f69a5
Packit 1f69a5
    case PA_SAMPLE_FLOAT32LE:
Packit 1f69a5
      return "F32LE";
Packit 1f69a5
Packit 1f69a5
    case PA_SAMPLE_FLOAT32BE:
Packit 1f69a5
      return "F32BE";
Packit 1f69a5
Packit 1f69a5
    case PA_SAMPLE_S32LE:
Packit 1f69a5
      return "S32LE";
Packit 1f69a5
Packit 1f69a5
    case PA_SAMPLE_S32BE:
Packit 1f69a5
      return "S32BE";
Packit 1f69a5
Packit 1f69a5
    case PA_SAMPLE_S24LE:
Packit 1f69a5
      return "S24LE";
Packit 1f69a5
Packit 1f69a5
    case PA_SAMPLE_S24BE:
Packit 1f69a5
      return "S24BE";
Packit 1f69a5
Packit 1f69a5
    case PA_SAMPLE_S24_32LE:
Packit 1f69a5
      return "S24_32LE";
Packit 1f69a5
Packit 1f69a5
    case PA_SAMPLE_S24_32BE:
Packit 1f69a5
      return "S24_32BE";
Packit 1f69a5
Packit 1f69a5
    default:
Packit 1f69a5
      return NULL;
Packit 1f69a5
  }
Packit 1f69a5
}
Packit 1f69a5
Packit 1f69a5
/* PATH_MAX is not defined everywhere, e.g. on GNU Hurd */
Packit 1f69a5
#ifndef PATH_MAX
Packit 1f69a5
#define PATH_MAX 4096
Packit 1f69a5
#endif
Packit 1f69a5
Packit 1f69a5
gchar *
Packit 1f69a5
gst_pulse_client_name (void)
Packit 1f69a5
{
Packit 1f69a5
  gchar buf[PATH_MAX];
Packit 1f69a5
Packit 1f69a5
  const char *c;
Packit 1f69a5
Packit 1f69a5
  if ((c = g_get_application_name ()))
Packit 1f69a5
    return g_strdup (c);
Packit 1f69a5
  else if (pa_get_binary_name (buf, sizeof (buf)))
Packit 1f69a5
    return g_strdup (buf);
Packit 1f69a5
  else
Packit 1f69a5
    return g_strdup_printf ("GStreamer-pid-%lu", (gulong) getpid ());
Packit 1f69a5
}
Packit 1f69a5
Packit 1f69a5
pa_channel_map *
Packit 1f69a5
gst_pulse_gst_to_channel_map (pa_channel_map * map,
Packit 1f69a5
    const GstAudioRingBufferSpec * spec)
Packit 1f69a5
{
Packit 1f69a5
  gint i, j;
Packit 1f69a5
  gint channels;
Packit 1f69a5
  const GstAudioChannelPosition *pos;
Packit 1f69a5
Packit 1f69a5
  pa_channel_map_init (map);
Packit 1f69a5
Packit 1f69a5
  channels = GST_AUDIO_INFO_CHANNELS (&spec->info);
Packit 1f69a5
  pos = spec->info.position;
Packit 1f69a5
Packit 1f69a5
  for (j = 0; j < channels; j++) {
Packit 1f69a5
    for (i = 0; i < G_N_ELEMENTS (gst_pa_pos_table); i++) {
Packit 1f69a5
      if (pos[j] == gst_pa_pos_table[i].gst_pos) {
Packit 1f69a5
        map->map[j] = gst_pa_pos_table[i].pa_pos;
Packit 1f69a5
        break;
Packit 1f69a5
      }
Packit 1f69a5
    }
Packit 1f69a5
    if (i == G_N_ELEMENTS (gst_pa_pos_table))
Packit 1f69a5
      return NULL;
Packit 1f69a5
  }
Packit 1f69a5
Packit 1f69a5
  if (j != spec->info.channels) {
Packit 1f69a5
    return NULL;
Packit 1f69a5
  }
Packit 1f69a5
Packit 1f69a5
  map->channels = spec->info.channels;
Packit 1f69a5
Packit 1f69a5
  if (!pa_channel_map_valid (map)) {
Packit 1f69a5
    return NULL;
Packit 1f69a5
  }
Packit 1f69a5
Packit 1f69a5
  return map;
Packit 1f69a5
}
Packit 1f69a5
Packit 1f69a5
GstAudioRingBufferSpec *
Packit 1f69a5
gst_pulse_channel_map_to_gst (const pa_channel_map * map,
Packit 1f69a5
    GstAudioRingBufferSpec * spec)
Packit 1f69a5
{
Packit 1f69a5
  gint i, j;
Packit 1f69a5
  gboolean invalid = FALSE;
Packit 1f69a5
  gint channels;
Packit 1f69a5
  GstAudioChannelPosition *pos;
Packit 1f69a5
Packit 1f69a5
  channels = GST_AUDIO_INFO_CHANNELS (&spec->info);
Packit 1f69a5
Packit 1f69a5
  g_return_val_if_fail (map->channels == channels, NULL);
Packit 1f69a5
Packit 1f69a5
  pos = spec->info.position;
Packit 1f69a5
Packit 1f69a5
  for (j = 0; j < channels; j++) {
Packit 1f69a5
    for (i = 0; j < channels && i < G_N_ELEMENTS (gst_pa_pos_table); i++) {
Packit 1f69a5
      if (map->map[j] == gst_pa_pos_table[i].pa_pos) {
Packit 1f69a5
        pos[j] = gst_pa_pos_table[i].gst_pos;
Packit 1f69a5
        break;
Packit 1f69a5
      }
Packit 1f69a5
    }
Packit 1f69a5
    if (i == G_N_ELEMENTS (gst_pa_pos_table))
Packit 1f69a5
      return NULL;
Packit 1f69a5
  }
Packit 1f69a5
Packit 1f69a5
  if (!invalid
Packit 1f69a5
      && !gst_audio_check_valid_channel_positions (pos, channels, FALSE))
Packit 1f69a5
    invalid = TRUE;
Packit 1f69a5
Packit 1f69a5
  if (invalid) {
Packit 1f69a5
    for (i = 0; i < channels; i++)
Packit 1f69a5
      pos[i] = GST_AUDIO_CHANNEL_POSITION_NONE;
Packit 1f69a5
  } else {
Packit 1f69a5
    if (pos[0] != GST_AUDIO_CHANNEL_POSITION_NONE)
Packit 1f69a5
      spec->info.flags &= ~GST_AUDIO_FLAG_UNPOSITIONED;
Packit 1f69a5
  }
Packit 1f69a5
Packit 1f69a5
  return spec;
Packit 1f69a5
}
Packit 1f69a5
Packit 1f69a5
void
Packit 1f69a5
gst_pulse_cvolume_from_linear (pa_cvolume * v, unsigned channels,
Packit 1f69a5
    gdouble volume)
Packit 1f69a5
{
Packit 1f69a5
  pa_cvolume_set (v, channels, pa_sw_volume_from_linear (volume));
Packit 1f69a5
}
Packit 1f69a5
Packit 1f69a5
static gboolean
Packit 1f69a5
make_proplist_item (GQuark field_id, const GValue * value, gpointer user_data)
Packit 1f69a5
{
Packit 1f69a5
  pa_proplist *p = (pa_proplist *) user_data;
Packit 1f69a5
  gchar *prop_id = (gchar *) g_quark_to_string (field_id);
Packit 1f69a5
Packit 1f69a5
  /* http://0pointer.de/lennart/projects/pulseaudio/doxygen/proplist_8h.html */
Packit 1f69a5
Packit 1f69a5
  /* match prop id */
Packit 1f69a5
Packit 1f69a5
  /* check type */
Packit 1f69a5
  switch (G_VALUE_TYPE (value)) {
Packit 1f69a5
    case G_TYPE_STRING:
Packit 1f69a5
      pa_proplist_sets (p, prop_id, g_value_get_string (value));
Packit 1f69a5
      break;
Packit 1f69a5
    default:
Packit 1f69a5
      GST_WARNING ("unmapped property type %s", G_VALUE_TYPE_NAME (value));
Packit 1f69a5
      break;
Packit 1f69a5
  }
Packit 1f69a5
Packit 1f69a5
  return TRUE;
Packit 1f69a5
}
Packit 1f69a5
Packit 1f69a5
pa_proplist *
Packit 1f69a5
gst_pulse_make_proplist (const GstStructure * properties)
Packit 1f69a5
{
Packit 1f69a5
  pa_proplist *proplist = pa_proplist_new ();
Packit 1f69a5
Packit 1f69a5
  /* iterate the structure and fill the proplist */
Packit 1f69a5
  gst_structure_foreach (properties, make_proplist_item, proplist);
Packit 1f69a5
  return proplist;
Packit 1f69a5
}
Packit 1f69a5
Packit 1f69a5
GstStructure *
Packit 1f69a5
gst_pulse_make_structure (pa_proplist * properties)
Packit 1f69a5
{
Packit 1f69a5
  GstStructure *str;
Packit 1f69a5
  void *state = NULL;
Packit 1f69a5
Packit 1f69a5
  str = gst_structure_new_empty ("pulse-proplist");
Packit 1f69a5
Packit 1f69a5
  while (TRUE) {
Packit 1f69a5
    const char *key, *val;
Packit 1f69a5
Packit 1f69a5
    key = pa_proplist_iterate (properties, &state);
Packit 1f69a5
    if (key == NULL)
Packit 1f69a5
      break;
Packit 1f69a5
Packit 1f69a5
    val = pa_proplist_gets (properties, key);
Packit 1f69a5
Packit 1f69a5
    gst_structure_set (str, key, G_TYPE_STRING, val, NULL);
Packit 1f69a5
  }
Packit 1f69a5
  return str;
Packit 1f69a5
}
Packit 1f69a5
Packit 1f69a5
static gboolean
Packit 1f69a5
gst_pulse_format_info_int_prop_to_value (pa_format_info * format,
Packit 1f69a5
    const char *key, GValue * value)
Packit 1f69a5
{
Packit 1f69a5
  GValue v = { 0, };
Packit 1f69a5
  int i;
Packit 1f69a5
  int *a, n;
Packit 1f69a5
  int min, max;
Packit 1f69a5
Packit 1f69a5
  if (pa_format_info_get_prop_int (format, key, &i) == 0) {
Packit 1f69a5
    g_value_init (value, G_TYPE_INT);
Packit 1f69a5
    g_value_set_int (value, i);
Packit 1f69a5
Packit 1f69a5
  } else if (pa_format_info_get_prop_int_array (format, key, &a, &n) == 0) {
Packit 1f69a5
    g_value_init (value, GST_TYPE_LIST);
Packit 1f69a5
    g_value_init (&v, G_TYPE_INT);
Packit 1f69a5
Packit 1f69a5
    for (i = 0; i < n; i++) {
Packit 1f69a5
      g_value_set_int (&v, a[i]);
Packit 1f69a5
      gst_value_list_append_value (value, &v);
Packit 1f69a5
    }
Packit 1f69a5
Packit 1f69a5
    pa_xfree (a);
Packit 1f69a5
Packit 1f69a5
  } else if (pa_format_info_get_prop_int_range (format, key, &min, &max) == 0) {
Packit 1f69a5
    g_value_init (value, GST_TYPE_INT_RANGE);
Packit 1f69a5
    gst_value_set_int_range (value, min, max);
Packit 1f69a5
Packit 1f69a5
  } else {
Packit 1f69a5
    /* Property not available or is not an int type */
Packit 1f69a5
    return FALSE;
Packit 1f69a5
  }
Packit 1f69a5
Packit 1f69a5
  return TRUE;
Packit 1f69a5
}
Packit 1f69a5
Packit 1f69a5
GstCaps *
Packit 1f69a5
gst_pulse_format_info_to_caps (pa_format_info * format)
Packit 1f69a5
{
Packit 1f69a5
  GstCaps *ret = NULL;
Packit 1f69a5
  GValue v = { 0, };
Packit 1f69a5
  pa_sample_spec ss;
Packit 1f69a5
Packit 1f69a5
  switch (format->encoding) {
Packit 1f69a5
    case PA_ENCODING_PCM:{
Packit 1f69a5
      char *tmp = NULL;
Packit 1f69a5
Packit 1f69a5
      pa_format_info_to_sample_spec (format, &ss, NULL);
Packit 1f69a5
Packit 1f69a5
      if (pa_format_info_get_prop_string (format,
Packit 1f69a5
              PA_PROP_FORMAT_SAMPLE_FORMAT, &tmp)) {
Packit 1f69a5
        /* No specific sample format means any sample format */
Packit 1f69a5
        ret = gst_caps_from_string (_PULSE_CAPS_PCM);
Packit 1f69a5
        goto out;
Packit 1f69a5
Packit 1f69a5
      } else if (ss.format == PA_SAMPLE_ALAW) {
Packit 1f69a5
        ret = gst_caps_from_string (_PULSE_CAPS_ALAW);
Packit 1f69a5
Packit 1f69a5
      } else if (ss.format == PA_SAMPLE_ULAW) {
Packit 1f69a5
        ret = gst_caps_from_string (_PULSE_CAPS_MULAW);
Packit 1f69a5
Packit 1f69a5
      } else {
Packit 1f69a5
        /* Linear PCM format */
Packit 1f69a5
        const char *sformat =
Packit 1f69a5
            gst_pulse_sample_format_to_caps_format (ss.format);
Packit 1f69a5
Packit 1f69a5
        ret = gst_caps_from_string (_PULSE_CAPS_LINEAR);
Packit 1f69a5
Packit 1f69a5
        if (sformat)
Packit 1f69a5
          gst_caps_set_simple (ret, "format", G_TYPE_STRING, NULL);
Packit 1f69a5
      }
Packit 1f69a5
Packit 1f69a5
      pa_xfree (tmp);
Packit 1f69a5
      break;
Packit 1f69a5
    }
Packit 1f69a5
Packit 1f69a5
    case PA_ENCODING_AC3_IEC61937:
Packit 1f69a5
      ret = gst_caps_from_string (_PULSE_CAPS_AC3);
Packit 1f69a5
      break;
Packit 1f69a5
Packit 1f69a5
    case PA_ENCODING_EAC3_IEC61937:
Packit 1f69a5
      ret = gst_caps_from_string (_PULSE_CAPS_EAC3);
Packit 1f69a5
      break;
Packit 1f69a5
Packit 1f69a5
    case PA_ENCODING_DTS_IEC61937:
Packit 1f69a5
      ret = gst_caps_from_string (_PULSE_CAPS_DTS);
Packit 1f69a5
      break;
Packit 1f69a5
Packit 1f69a5
    case PA_ENCODING_MPEG_IEC61937:
Packit 1f69a5
      ret = gst_caps_from_string (_PULSE_CAPS_MP3);
Packit 1f69a5
      break;
Packit 1f69a5
Packit 1f69a5
    default:
Packit 1f69a5
      GST_WARNING ("Found a PA format that we don't support yet");
Packit 1f69a5
      goto out;
Packit 1f69a5
  }
Packit 1f69a5
Packit 1f69a5
  if (gst_pulse_format_info_int_prop_to_value (format, PA_PROP_FORMAT_RATE, &v))
Packit 1f69a5
    gst_caps_set_value (ret, "rate", &v);
Packit 1f69a5
Packit 1f69a5
  g_value_unset (&v);
Packit 1f69a5
Packit 1f69a5
  if (gst_pulse_format_info_int_prop_to_value (format, PA_PROP_FORMAT_CHANNELS,
Packit 1f69a5
          &v))
Packit 1f69a5
    gst_caps_set_value (ret, "channels", &v);
Packit 1f69a5
Packit 1f69a5
out:
Packit 1f69a5
  return ret;
Packit 1f69a5
}