sndio_dec.c 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117
  1. /*
  2. * sndio play and grab interface
  3. * Copyright (c) 2010 Jacob Meuser
  4. *
  5. * This file is part of FFmpeg.
  6. *
  7. * FFmpeg 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. * FFmpeg 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 FFmpeg; 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 "libavformat/internal.h"
  25. #include "libavutil/opt.h"
  26. #include "sndio_common.h"
  27. static av_cold int audio_read_header(AVFormatContext *s1)
  28. {
  29. SndioData *s = s1->priv_data;
  30. AVStream *st;
  31. int ret;
  32. st = avformat_new_stream(s1, NULL);
  33. if (!st)
  34. return AVERROR(ENOMEM);
  35. ret = ff_sndio_open(s1, 0, s1->filename);
  36. if (ret < 0)
  37. return ret;
  38. /* take real parameters */
  39. st->codec->codec_type = AVMEDIA_TYPE_AUDIO;
  40. st->codec->codec_id = s->codec_id;
  41. st->codec->sample_rate = s->sample_rate;
  42. st->codec->channels = s->channels;
  43. avpriv_set_pts_info(st, 64, 1, 1000000); /* 64 bits pts in us */
  44. return 0;
  45. }
  46. static int audio_read_packet(AVFormatContext *s1, AVPacket *pkt)
  47. {
  48. SndioData *s = s1->priv_data;
  49. int64_t bdelay, cur_time;
  50. int ret;
  51. if ((ret = av_new_packet(pkt, s->buffer_size)) < 0)
  52. return ret;
  53. ret = sio_read(s->hdl, pkt->data, pkt->size);
  54. if (ret == 0 || sio_eof(s->hdl)) {
  55. av_free_packet(pkt);
  56. return AVERROR_EOF;
  57. }
  58. pkt->size = ret;
  59. s->softpos += ret;
  60. /* compute pts of the start of the packet */
  61. cur_time = av_gettime();
  62. bdelay = ret + s->hwpos - s->softpos;
  63. /* convert to pts */
  64. pkt->pts = cur_time - ((bdelay * 1000000) /
  65. (s->bps * s->channels * s->sample_rate));
  66. return 0;
  67. }
  68. static av_cold int audio_read_close(AVFormatContext *s1)
  69. {
  70. SndioData *s = s1->priv_data;
  71. ff_sndio_close(s);
  72. return 0;
  73. }
  74. static const AVOption options[] = {
  75. { "sample_rate", "", offsetof(SndioData, sample_rate), AV_OPT_TYPE_INT, {.dbl = 48000}, 1, INT_MAX, AV_OPT_FLAG_DECODING_PARAM },
  76. { "channels", "", offsetof(SndioData, channels), AV_OPT_TYPE_INT, {.dbl = 2}, 1, INT_MAX, AV_OPT_FLAG_DECODING_PARAM },
  77. { NULL },
  78. };
  79. static const AVClass sndio_demuxer_class = {
  80. .class_name = "sndio indev",
  81. .item_name = av_default_item_name,
  82. .option = options,
  83. .version = LIBAVUTIL_VERSION_INT,
  84. };
  85. AVInputFormat ff_sndio_demuxer = {
  86. .name = "sndio",
  87. .long_name = NULL_IF_CONFIG_SMALL("sndio audio capture"),
  88. .priv_data_size = sizeof(SndioData),
  89. .read_header = audio_read_header,
  90. .read_packet = audio_read_packet,
  91. .read_close = audio_read_close,
  92. .flags = AVFMT_NOFILE,
  93. .priv_class = &sndio_demuxer_class,
  94. };