sndio_dec.c 3.4 KB

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