You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

123 lines
3.4KB

  1. /*
  2. * sndio play and grab interface
  3. * Copyright (c) 2010 Jacob Meuser
  4. *
  5. * This file is part of Libav.
  6. *
  7. * Libav is free software; you can redistribute it and/or
  8. * modify it under the terms of the GNU Lesser General Public
  9. * License as published by the Free Software Foundation; either
  10. * version 2.1 of the License, or (at your option) any later version.
  11. *
  12. * Libav is distributed in the hope that it will be useful,
  13. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  15. * Lesser General Public License for more details.
  16. *
  17. * You should have received a copy of the GNU Lesser General Public
  18. * License along with Libav; if not, write to the Free Software
  19. * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
  20. */
  21. #include <stdint.h>
  22. #include <sndio.h>
  23. #include "libavformat/avformat.h"
  24. #include "libavutil/opt.h"
  25. #include "sndio_common.h"
  26. static av_cold int audio_read_header(AVFormatContext *s1,
  27. AVFormatParameters *ap)
  28. {
  29. SndioData *s = s1->priv_data;
  30. AVStream *st;
  31. int ret;
  32. if (ap->sample_rate > 0)
  33. s->sample_rate = ap->sample_rate;
  34. if (ap->channels > 0)
  35. s->channels = ap->channels;
  36. st = av_new_stream(s1, 0);
  37. if (!st)
  38. return AVERROR(ENOMEM);
  39. ret = ff_sndio_open(s1, 0, s1->filename);
  40. if (ret < 0)
  41. return ret;
  42. /* take real parameters */
  43. st->codec->codec_type = AVMEDIA_TYPE_AUDIO;
  44. st->codec->codec_id = s->codec_id;
  45. st->codec->sample_rate = s->sample_rate;
  46. st->codec->channels = s->channels;
  47. av_set_pts_info(st, 64, 1, 1000000); /* 64 bits pts in us */
  48. return 0;
  49. }
  50. static int audio_read_packet(AVFormatContext *s1, AVPacket *pkt)
  51. {
  52. SndioData *s = s1->priv_data;
  53. int64_t bdelay, cur_time;
  54. int ret;
  55. if ((ret = av_new_packet(pkt, s->buffer_size)) < 0)
  56. return ret;
  57. ret = sio_read(s->hdl, pkt->data, pkt->size);
  58. if (ret == 0 || sio_eof(s->hdl)) {
  59. av_free_packet(pkt);
  60. return AVERROR_EOF;
  61. }
  62. pkt->size = ret;
  63. s->softpos += ret;
  64. /* compute pts of the start of the packet */
  65. cur_time = av_gettime();
  66. bdelay = ret + s->hwpos - s->softpos;
  67. /* convert to pts */
  68. pkt->pts = cur_time - ((bdelay * 1000000) /
  69. (s->bps * s->channels * s->sample_rate));
  70. return 0;
  71. }
  72. static av_cold int audio_read_close(AVFormatContext *s1)
  73. {
  74. SndioData *s = s1->priv_data;
  75. ff_sndio_close(s);
  76. return 0;
  77. }
  78. static const AVOption options[] = {
  79. { "sample_rate", "", offsetof(SndioData, sample_rate), FF_OPT_TYPE_INT, {.dbl = 48000}, 1, INT_MAX, AV_OPT_FLAG_DECODING_PARAM },
  80. { "channels", "", offsetof(SndioData, channels), FF_OPT_TYPE_INT, {.dbl = 2}, 1, INT_MAX, AV_OPT_FLAG_DECODING_PARAM },
  81. { NULL },
  82. };
  83. static const AVClass sndio_demuxer_class = {
  84. .class_name = "sndio indev",
  85. .item_name = av_default_item_name,
  86. .option = options,
  87. .version = LIBAVUTIL_VERSION_INT,
  88. };
  89. AVInputFormat ff_sndio_demuxer = {
  90. .name = "sndio",
  91. .long_name = NULL_IF_CONFIG_SMALL("sndio audio capture"),
  92. .priv_data_size = sizeof(SndioData),
  93. .read_header = audio_read_header,
  94. .read_packet = audio_read_packet,
  95. .read_close = audio_read_close,
  96. .flags = AVFMT_NOFILE,
  97. .priv_class = &sndio_demuxer_class,
  98. };