FFmpeg之yuv镜像(十八)

for (int w = 0; w < lineDataSize; w += 3) {

mirrorBgr24[lineStartIndex + w] = bgr24[lineStartIndex + lineDataSize - w - 3];

mirrorBgr24[lineStartIndex + w + 1] = bgr24[lineStartIndex + lineDataSize - w - 2];

mirrorBgr24[lineStartIndex + w + 2] = bgr24[lineStartIndex + lineDataSize - w - 1];

}

lineStartIndex += lineDataSize;

}

}

  • 垂直镜像

void verticalMirrorBgr24(char *bgr24, char *mirrorBgr24, int width, int height) {

int lineDataSize = width * 3;

bgr24 += width * height * 3;

for (int h = 0; h < height; h++) {

memcpy(mirrorBgr24, bgr24, lineDataSize);

mirrorBgr24 += lineDataSize;

bgr24 -= lineDataSize;

}

}

三、镜像NV21 / NV12

由于NV21和NV12的结构只是UV数据位置相反,因此镜像NV21的代码同样适用于镜像NV12。

  • 水平镜像

void horizontalMirrorNv21(char *nv21, char *mirrorNv21, int width, int height) {

int yLineStartIndex = 0;

int uvLineStartIndex = width * height;

for (int h = 0; h < height; h++) {

for (int w = 0; w < width; w += 2) {

mirrorNv21[yLineStartIndex + w] = nv21[yLineStartIndex + width - w];

mirrorNv21[yLineStartIndex + w + 1] = nv21[yLineStartIndex + width - w - 1];

if ((h & 1) == 0) {

mirrorNv21[uvLineStartIndex + w] = nv21[uvLineStartIndex + width - w];

mirrorNv21[uvLineStartIndex + w + 1] = nv21[uvLineStartIndex + width - w - 1];

}

}

yLineStartIndex += width;

if ((h & 1) == 0) {

uvLineStartIndex += width;

}

}

}

  • 垂直镜像

void verticalMirrorNv21(char *nv21, char *mirrorNv21, int width, int height) {

int yLineDataSize = width;

int nv21Index = 0, mirrorNv21Index = 0;

nv21Index += width * height - yLineDataSize;

//mirror y

for (int h = 0; h < height; h++) {

memcpy(mirrorNv21 + mirrorNv21Index, nv21 + nv21Index, yLineDataSize);

mirrorNv21Index += yLineDataSize;

nv21Index -= yLineDataSize;

}

int uvLineDataSize = width;

//mirror uv

int uvHeight = height / 2;

//point to final line of uv

nv21Index += (width * height * 3 / 2 - uvLineDataSize);

for (int h = 0; h < uvHeight; h++) {

memcpy(mirrorNv21 + mirrorNv21Index, nv21 + nv21Index, uvLineDataSize);

mirrorNv21Index += uvLineDataSize;

nv21Index -= uvLineDataSize;

}

}

四、镜像I420 / YV12

由于I420 和YV12的结构只是UV数据位置相反,因此镜像I420的代码同样适用于镜像YV12。

  • 水平镜像

void horizontalMirrorI420(char *i420, char *mirrorI420, int width, int height) {

int yLineStartIndex = 0;

int uLineStartIndex = width * height;

int vLineStartIndex = width * height * 5 / 4;

for (int h = 0; h < height; h++) {

for (int w = 0; w < width; w += 2) {

mirrorI420[yLineStartIndex + w] = i420[yLineStartIndex + width - w];

mirrorI420[yLineStartIndex + w + 1] = i420[yLineStartIndex + width - w - 1];

if ((h & 1) == 0) {

mirrorI420[uLineStartIndex + (w >> 1)] = i420[uLineStartIndex + ((width - w) >> 1)];

mirrorI420[vLineStartIndex + (w >> 1) + 1] = i420[vLineStartIndex +

((width - w) >> 1) - 1];

}

}

yLineStartIndex += width;

if ((h & 1) == 0) {

uLineStartIndex += width >> 1;

vLineStartIndex += width >> 1;

}

}

}

  • 垂直镜像

void verticalMirrorI420(char *i420, char *mirrorI420, int width, int height) {

int halfWidth = width / 2;

int yLineDataSize = width;

int i420Index = 0, mirrorI420Index = 0;

i420Index += width * height - yLineDataSize;

//mirror y

for (int h = 0; h < height; h++) {

memcpy(mirrorI420 + mirrorI420Index, i420 + i420Index, yLineDataSize);

mirrorI420Index += yLineDataSize;

i420Index -= yLineDataSize;

}

int uvLineDataSize = halfWidth;

//mirror uv

int uHeight = height / 2;

//point to final line of u

i420Index += (width * height * 5 / 4 + yLineDataSize - uvLineDataSize);

for (int h = 0; h < uHeight; h++) {

memcpy(mirrorI420 + mirrorI420Index, i420 + i420Index, uvLineDataSize);

mirrorI420Index += uvLineDataSize;

i420Index -= uvLineDataSize;

}

//point to final line of v

i420Index += width * height / 2;

int vHeight = height / 2;

for (int h = 0; h < vHeight; h++) {

memcpy(mirrorI420 + mirrorI420Index, i420 + i420Index, uvLineDataSize);

mirrorI420Index += uvLineDataSize;

i420Index -= uvLineDataSize;

}

}

五、镜像YUYV

YUYV的共用关系是每2个Y共用1组U和V,因此在水平镜像时需要每4个Y替换其中两个Y的顺序。而垂直镜像时,整行处理即可。

  • 水平镜像

void horizontalMirrorYuyv(char *yuyv, char *mirrorYuyv, int width, int height) {

int lineStartIndex = 0;

int lineDataSize = width * 2;

for (int h = 0; h < height; h++) {

for (int w = 0; w < lineDataSize; w += 4) {

mirrorYuyv[lineStartIndex + w] = yuyv[lineStartIndex + lineDataSize - w - 2];

mirrorYuyv[lineStartIndex + w + 1] = yuyv[lineStartIndex + lineDataSize - w - 3];

mirrorYuyv[lineStartIndex + w + 2] = yuyv[lineStartIndex + lineDataSize - w - 4];

mirrorYuyv[lineStartIndex + w + 3] = yuyv[lineStartIndex + lineDataSize - w - 1];

}

lineStartIndex += lineDataSize;

}

}

学习福利

【Android 详细知识点思维脑图(技能树)】

其实Android开发的知识点就那么多,面试问来问去还是那么点东西。所以面试没有其他的诀窍,只看你对这些知识点准备的充分程度。so,出去面试时先看看自己复习到了哪个阶段就好。

虽然 Android 没有前几年火热了,已经过去了会四大组件就能找到高薪职位的时代了。这只能说明 Android 中级以下的岗位饱和了,现在高级工程师还是比较缺少的,很多高级职位给的薪资真的特别高(钱多也不一定能找到合适的),所以努力让自己成为高级工程师才是最重要的。

这里附上上述的面试题相关的几十套字节跳动,京东,小米,腾讯、头条、阿里、美团等公司19年的面试题。把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节。

由于篇幅有限,这里以图片的形式给大家展示一小部分。

网上学习 Android的资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。希望这份系统化的技术体系对大家有一个方向参考。
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!

这里附上上述的面试题相关的几十套字节跳动,京东,小米,腾讯、头条、阿里、美团等公司19年的面试题。把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节。

由于篇幅有限,这里以图片的形式给大家展示一小部分。

[外链图片转存中…(img-dqTS74us-1714956251090)]

网上学习 Android的资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。希望这份系统化的技术体系对大家有一个方向参考。
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值