记录一个`ffmpeg`的`swscale`库crash的例子

记录一个ffmpegswscale库crash的例子

本质上, 就是simd越界导致, 具体场景如下:

外部api分配了rgb24内存指针ptr, 然后转换为yuv420p, 如果ptrend不可访问, 则会crash

1. simd越界例子

cpp 复制代码
#include <cstdint>
#include <immintrin.h>
#include <iostream>
#include <memory>

#ifdef _WIN32
#include <windows.h>
#elif __linux__
#include <sys/mman.h>
#include <unistd.h>
#endif

#ifdef _WIN32
void win_crash() {
  constexpr size_t N = 1024;
  constexpr size_t PAGE = 4096;
  std::shared_ptr<uint8_t> base(
      static_cast<uint8_t *>(VirtualAlloc(
          nullptr, PAGE * 2, MEM_COMMIT | MEM_RESERVE, PAGE_READWRITE)),
      [](uint8_t *p) { VirtualFree(p, 0, MEM_RELEASE); });
  DWORD old;
  VirtualProtect(base.get() + PAGE, PAGE, PAGE_NOACCESS, &old);
  uint8_t *data = base.get() + PAGE - N;
  for (size_t i = 0; i < N; ++i)
    data[i] = static_cast<uint8_t>(i);

  for (size_t i = 0; i < N; i += 8) {
    __m256i v =
        _mm256_loadu_si256(reinterpret_cast<const __m256i *>(data + 32 + i));
    uint8_t tmp = static_cast<uint8_t>(_mm256_extract_epi8(v, 0));
    std::cout << static_cast<int64_t>(tmp) << "\n";
  }
}
#endif

#ifdef __linux__
// target_compile_options(my_target_name PRIVATE -mavx2)
void linux_crash() {
  constexpr size_t N = 1024;
  constexpr size_t PAGE = 4096;
  void *p = mmap(nullptr, PAGE * 2, PROT_READ | PROT_WRITE,
                 MAP_PRIVATE | MAP_ANONYMOUS, -1, 0);
  if (p == MAP_FAILED) {
    perror("mmap");
    return;
  }
  std::shared_ptr<uint8_t> base(static_cast<uint8_t *>(p),
                                [](uint8_t *p) { munmap(p, PAGE * 2); });
  if (mprotect(base.get() + PAGE, PAGE, PROT_NONE) != 0) {
    perror("mprotect");
    return;
  }
  uint8_t *data = base.get() + PAGE - N;
  for (size_t i = 0; i < N; ++i)
    data[i] = static_cast<uint8_t>(i);
  for (size_t i = 0; i < N; i += 8) {
    __m256i v = _mm256_loadu_si256(reinterpret_cast<const __m256i *>(data + i));
    uint8_t tmp = static_cast<uint8_t>(_mm256_extract_epi8(v, 0));
    std::cout << static_cast<int64_t>(tmp) << "\n";
  }
}
#endif

int main() {
  win_crash();
  // linux_crash();
  return 0;
}

ffmpeg堆栈错误信息

最终崩溃在函数指针lumToYV12中, 具体汇编代码见libswscale/x86/swscale.clibswscale/input.asm

reference

相关推荐
luoqice1 天前
RTMP视频流的帧格式分析
网络·ffmpeg
老姚---老姚2 天前
编译支持HEVC/H.265 over RTMP / Enhanced RTMP 的 ffmpeg
ffmpeg·h.265·hevc·rtmp·enhanced
码流怪侠3 天前
FFmpeg 开发实战全解析:从入门到精通(附完整代码示例)
ffmpeg·音视频开发·视频编码
圆弧YH3 天前
FFmpeg
ffmpeg
luoqice3 天前
FLV文件格式详解
ffmpeg
happybasic4 天前
在CMD下使用FFmpeg将.wav文件转换成指定的格式~
ffmpeg
shao9185164 天前
第10章 Streaming(上):初级音频应用(1)——项目三:自建服务器的Mini-Omni实时语音聊天机器人
ffmpeg·whisper·asr·mini-omni·自建语音服务器
Leon_Chenl5 天前
【已开源】【嵌入式 Linux 音视频+ AI 实战项目】瑞芯微 Rockchip 系列 RK3588-基于深度学习的人脸门禁+ IPC 智能安防监控系统
深度学习·opencv·yolo·ffmpeg·音视频·边缘计算·人脸识别+检测
antzou5 天前
视频图片/文字水印
ffmpeg·视频水印·批量水印
AC赳赳老秦6 天前
DBA 专属方案:用 OpenClaw 实现 SQL 语句优化、慢查询分析、数据库备份巡检全自动化
服务器·前端·数据库·ffmpeg·自动化·deepseek·openclaw