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.

92 lines
2.7KB

  1. /*
  2. * WavPack muxer
  3. * Copyright (c) 2013 Konstantin Shishkov <kostya.shishkov@gmail.com>
  4. * Copyright (c) 2012 Paul B Mahol
  5. *
  6. * This file is part of FFmpeg.
  7. *
  8. * FFmpeg is free software; you can redistribute it and/or
  9. * modify it under the terms of the GNU Lesser General Public
  10. * License as published by the Free Software Foundation; either
  11. * version 2.1 of the License, or (at your option) any later version.
  12. *
  13. * FFmpeg is distributed in the hope that it will be useful,
  14. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  15. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  16. * Lesser General Public License for more details.
  17. *
  18. * You should have received a copy of the GNU Lesser General Public
  19. * License along with FFmpeg; if not, write to the Free Software
  20. * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
  21. */
  22. #include "libavutil/attributes.h"
  23. #include "apetag.h"
  24. #include "avformat.h"
  25. #include "wv.h"
  26. typedef struct WvMuxContext {
  27. int64_t samples;
  28. } WvMuxContext;
  29. static av_cold int wv_write_header(AVFormatContext *ctx)
  30. {
  31. if (ctx->nb_streams > 1 ||
  32. ctx->streams[0]->codec->codec_id != AV_CODEC_ID_WAVPACK) {
  33. av_log(ctx, AV_LOG_ERROR, "This muxer only supports a single WavPack stream.\n");
  34. return AVERROR(EINVAL);
  35. }
  36. return 0;
  37. }
  38. static int wv_write_packet(AVFormatContext *ctx, AVPacket *pkt)
  39. {
  40. WvMuxContext *s = ctx->priv_data;
  41. WvHeader header;
  42. int ret;
  43. if (pkt->size < WV_HEADER_SIZE ||
  44. (ret = ff_wv_parse_header(&header, pkt->data)) < 0) {
  45. av_log(ctx, AV_LOG_ERROR, "Invalid WavPack packet.\n");
  46. return AVERROR(EINVAL);
  47. }
  48. s->samples += header.samples;
  49. avio_write(ctx->pb, pkt->data, pkt->size);
  50. return 0;
  51. }
  52. static av_cold int wv_write_trailer(AVFormatContext *ctx)
  53. {
  54. WvMuxContext *s = ctx->priv_data;
  55. /* update total number of samples in the first block */
  56. if (ctx->pb->seekable && s->samples &&
  57. s->samples < UINT32_MAX) {
  58. int64_t pos = avio_tell(ctx->pb);
  59. avio_seek(ctx->pb, 12, SEEK_SET);
  60. avio_wl32(ctx->pb, s->samples);
  61. avio_seek(ctx->pb, pos, SEEK_SET);
  62. }
  63. ff_ape_write_tag(ctx);
  64. return 0;
  65. }
  66. AVOutputFormat ff_wv_muxer = {
  67. .name = "wv",
  68. .long_name = NULL_IF_CONFIG_SMALL("raw WavPack"),
  69. .mime_type = "audio/x-wavpack",
  70. .extensions = "wv",
  71. .priv_data_size = sizeof(WvMuxContext),
  72. .audio_codec = AV_CODEC_ID_WAVPACK,
  73. .video_codec = AV_CODEC_ID_NONE,
  74. .write_header = wv_write_header,
  75. .write_packet = wv_write_packet,
  76. .write_trailer = wv_write_trailer,
  77. .flags = AVFMT_NOTIMESTAMPS,
  78. };