hwc: release fence in perform_fimc

Change-Id: I2039939c42a76567874ec0e1100ab1cbff95ed39
This commit is contained in:
Shilin Victor 2019-01-26 23:52:27 +03:00
parent e24564745e
commit aa2cfeb6f6

View file

@ -790,12 +790,22 @@ static int perform_fimg(hwc_context_t *ctx, const hwc_layer_1_t &layer, struct h
} }
return ret; return ret;
} }
static int perform_fimc(hwc_context_t *ctx, const hwc_layer_1_t &layer, struct hwc_win_info_t &win) static int perform_fimc(hwc_context_t *ctx, const hwc_layer_1_t &layer, struct hwc_win_info_t &win)
{ {
private_handle_t *src_handle = private_handle_t::dynamicCast(layer.handle); private_handle_t *src_handle = private_handle_t::dynamicCast(layer.handle);
hwc_rect_t crop;
int ret = 0; int ret = 0;
int rotate;
struct private_handle_t *dst_handle = private_handle_t::dynamicCast(win.dst_buf[win.current_buf]);
uint32_t dst_addr = (uint32_t) dst_handle->paddr;
int w, h;
struct fimc_buf dst_buf;
struct fimc_buf buf;
// before sending anything to FIMC // before sending anything to FIMC
if (layer.acquireFenceFd >= 0) { if (layer.acquireFenceFd >= 0) {
@ -805,60 +815,59 @@ static int perform_fimc(hwc_context_t *ctx, const hwc_layer_1_t &layer, struct h
ret = v4l2_reqbufs_out(ctx, 0); ret = v4l2_reqbufs_out(ctx, 0);
if (ret < 0) { if (ret < 0) {
ALOGE("%s: v4l2_reqbufs_out() rc=%d", __FUNCTION__, ret); ALOGE("%s: v4l2_reqbufs_out() rc=%d", __FUNCTION__, ret);
return ret; goto release;
} }
//src //src
hwc_rect_t crop = integerizeSourceCrop(layer.sourceCropf); crop = integerizeSourceCrop(layer.sourceCropf);
ret = v4l2_s_fmt_pix_out(ctx, EXYNOS4_ALIGN(src_handle->width, 16), EXYNOS4_ALIGN(src_handle->height, 16), HAL_PIXEL_FORMAT_2_V4L2_PIX(src_handle->format), 0); ret = v4l2_s_fmt_pix_out(ctx, EXYNOS4_ALIGN(src_handle->width, 16), EXYNOS4_ALIGN(src_handle->height, 16), HAL_PIXEL_FORMAT_2_V4L2_PIX(src_handle->format), 0);
if (ret < 0) { if (ret < 0) {
ALOGE("%s: v4l2_s_fmt_pix_out() rc=%d", __FUNCTION__, ret); ALOGE("%s: v4l2_s_fmt_pix_out() rc=%d", __FUNCTION__, ret);
return ret; goto release;
} }
//some yuv formats do need width and height to be multiple of 2 //some yuv formats do need width and height to be multiple of 2
ret = v4l2_s_crop(ctx, crop.left, crop.top, multipleOf2(WIDTH(crop)), multipleOf2(HEIGHT(crop))); ret = v4l2_s_crop(ctx, crop.left, crop.top, multipleOf2(WIDTH(crop)), multipleOf2(HEIGHT(crop)));
if (ret < 0) { if (ret < 0) {
ALOGE("%s: v4l2_s_crop() rc=%d", __FUNCTION__, ret); ALOGE("%s: v4l2_s_crop() rc=%d", __FUNCTION__, ret);
return ret; goto release;
} }
ret = v4l2_reqbufs_out(ctx, 1); ret = v4l2_reqbufs_out(ctx, 1);
if (ret < 0) { if (ret < 0) {
ALOGE("%s: v4l2_reqbufs_out() rc=%d", __FUNCTION__, ret); ALOGE("%s: v4l2_reqbufs_out() rc=%d", __FUNCTION__, ret);
return ret; goto release;
} }
//dst //dst
int rotate = rotateValueHAL2PP(layer.transform); rotate = rotateValueHAL2PP(layer.transform);
ret = v4l2_s_ctrl(ctx, V4L2_CID_ROTATION, rotate); ret = v4l2_s_ctrl(ctx, V4L2_CID_ROTATION, rotate);
if (ret < 0) { if (ret < 0) {
ALOGE("%s: s_ctrl V4L2_CID_ROTATION rc=%d", __FUNCTION__, ret); ALOGE("%s: s_ctrl V4L2_CID_ROTATION rc=%d", __FUNCTION__, ret);
return ret; goto release;
} }
ret = v4l2_s_ctrl(ctx, V4L2_CID_HFLIP, layer.transform == HWC_TRANSFORM_FLIP_H?1:0); ret = v4l2_s_ctrl(ctx, V4L2_CID_HFLIP, layer.transform == HWC_TRANSFORM_FLIP_H?1:0);
if (ret < 0) { if (ret < 0) {
ALOGE("%s: s_ctrl V4L2_CID_HFLIP rc=%d", __FUNCTION__, ret); ALOGE("%s: s_ctrl V4L2_CID_HFLIP rc=%d", __FUNCTION__, ret);
return ret; goto release;
} }
ret = v4l2_s_ctrl(ctx, V4L2_CID_VFLIP, layer.transform == HWC_TRANSFORM_FLIP_V?1:0); ret = v4l2_s_ctrl(ctx, V4L2_CID_VFLIP, layer.transform == HWC_TRANSFORM_FLIP_V?1:0);
if (ret < 0) { if (ret < 0) {
ALOGE("%s: s_ctrl V4L2_CID_VFLIP rc=%d", __FUNCTION__, ret); ALOGE("%s: s_ctrl V4L2_CID_VFLIP rc=%d", __FUNCTION__, ret);
return ret; goto release;
} }
ret = v4l2_g_fbuf(ctx, NULL, NULL, NULL, NULL); //TODO, check whether so it is ok ret = v4l2_g_fbuf(ctx, NULL, NULL, NULL, NULL); //TODO, check whether so it is ok
if (ret < 0) { if (ret < 0) {
ALOGE("%s: v4l2_g_fbuf() rc=%d", __FUNCTION__, ret); ALOGE("%s: v4l2_g_fbuf() rc=%d", __FUNCTION__, ret);
return ret; goto release;
} }
struct private_handle_t *dst_handle = private_handle_t::dynamicCast(win.dst_buf[win.current_buf]); dst_handle = private_handle_t::dynamicCast(win.dst_buf[win.current_buf]);
uint32_t dst_addr = (uint32_t) dst_handle->paddr; dst_addr = (uint32_t) dst_handle->paddr;
int w, h;
if (rotate == 90 || rotate == 270) { if (rotate == 90 || rotate == 270) {
w = HEIGHT(layer.displayFrame); w = HEIGHT(layer.displayFrame);
@ -871,35 +880,33 @@ static int perform_fimc(hwc_context_t *ctx, const hwc_layer_1_t &layer, struct h
ret = v4l2_s_fbuf(ctx, (void *)dst_addr, w, h, V4L2_PIX_FMT_RGB32); ret = v4l2_s_fbuf(ctx, (void *)dst_addr, w, h, V4L2_PIX_FMT_RGB32);
if (ret < 0) { if (ret < 0) {
ALOGE("%s: v4l2_s_fbuf(dst_addr(0x%x)) rc=%d", __FUNCTION__, dst_addr, ret); ALOGE("%s: v4l2_s_fbuf(dst_addr(0x%x)) rc=%d", __FUNCTION__, dst_addr, ret);
return ret; goto release;
} }
struct fimc_buf dst_buf;
memset(&dst_buf, 0, sizeof(struct fimc_buf)); memset(&dst_buf, 0, sizeof(struct fimc_buf));
dst_buf.base[FIMC_ADDR_Y] = dst_addr; dst_buf.base[FIMC_ADDR_Y] = dst_addr;
ret = v4l2_s_ctrl(ctx, V4L2_CID_DST_INFO, (int) &dst_buf); ret = v4l2_s_ctrl(ctx, V4L2_CID_DST_INFO, (int) &dst_buf);
if (ret < 0) { if (ret < 0) {
ALOGE("%s: s_ctrl V4L2_CID_DST_INFO rc=%d", __FUNCTION__, ret); ALOGE("%s: s_ctrl V4L2_CID_DST_INFO rc=%d", __FUNCTION__, ret);
return ret; goto release;
} }
ret = v4l2_s_fmt_win(ctx, 0, 0, w, h); ret = v4l2_s_fmt_win(ctx, 0, 0, w, h);
if (ret < 0) { if (ret < 0) {
ALOGE("%s: v4l2_s_fmt_win() rc=%d", __FUNCTION__, ret); ALOGE("%s: v4l2_s_fmt_win() rc=%d", __FUNCTION__, ret);
return ret; goto release;
} }
//oneshot //oneshot
ret = v4l2_streamon_out(ctx); ret = v4l2_streamon_out(ctx);
if (ret < 0) { if (ret < 0) {
ALOGE("%s: v4l2_streamon_out() rc=%d", __FUNCTION__, ret); ALOGE("%s: v4l2_streamon_out() rc=%d", __FUNCTION__, ret);
return ret; goto release;
} }
ALOGV("%s: src_handle usage(0x%x) base(0x%x) paddr(0x%x)", __FUNCTION__, src_handle->usage, src_handle->base, src_handle->paddr); ALOGV("%s: src_handle usage(0x%x) base(0x%x) paddr(0x%x)", __FUNCTION__, src_handle->usage, src_handle->base, src_handle->paddr);
struct fimc_buf buf;
buf.base[FIMC_ADDR_Y] = src_handle->paddr + src_handle->offset; buf.base[FIMC_ADDR_Y] = src_handle->paddr + src_handle->offset;
buf.base[FIMC_ADDR_CB] = src_handle->paddr + src_handle->offset + src_handle->uoffset; buf.base[FIMC_ADDR_CB] = src_handle->paddr + src_handle->offset + src_handle->uoffset;
buf.base[FIMC_ADDR_CR] = src_handle->paddr + src_handle->offset + src_handle->uoffset + src_handle->voffset; buf.base[FIMC_ADDR_CR] = src_handle->paddr + src_handle->offset + src_handle->uoffset + src_handle->voffset;
@ -907,20 +914,20 @@ static int perform_fimc(hwc_context_t *ctx, const hwc_layer_1_t &layer, struct h
ret = v4l2_qbuf(ctx, 0, (unsigned long) &buf); ret = v4l2_qbuf(ctx, 0, (unsigned long) &buf);
if (ret < 0) { if (ret < 0) {
ALOGE("%s: v4l2_qbuf(0x%x) rc=%d", __FUNCTION__, src_handle->base, ret); ALOGE("%s: v4l2_qbuf(0x%x) rc=%d", __FUNCTION__, src_handle->base, ret);
return ret; goto release;
} }
//wait?? //wait??
ret = v4l2_dqbuf(ctx); ret = v4l2_dqbuf(ctx);
if (ret < 0) { if (ret < 0) {
ALOGE("%s: v4l2_dqbuf() rc=%d", __FUNCTION__, ret); ALOGE("%s: v4l2_dqbuf() rc=%d", __FUNCTION__, ret);
return ret; goto release;
} }
ret = v4l2_streamoff(ctx); ret = v4l2_streamoff(ctx);
if (ret < 0) { if (ret < 0) {
ALOGE("%s: v4l2_streamoff() rc=%d", __FUNCTION__, ret); ALOGE("%s: v4l2_streamoff() rc=%d", __FUNCTION__, ret);
return ret; goto release;
} }
if (layer.acquireFenceFd >= 0) { if (layer.acquireFenceFd >= 0) {
@ -928,6 +935,12 @@ static int perform_fimc(hwc_context_t *ctx, const hwc_layer_1_t &layer, struct h
} }
return w; return w;
release:
if (layer.acquireFenceFd >= 0) {
close(layer.acquireFenceFd);
}
return ret;
} }
static void config_overlay(hwc_context_t *ctx, hwc_layer_1_t &layer, s3c_fb_win_config &cfg) static void config_overlay(hwc_context_t *ctx, hwc_layer_1_t &layer, s3c_fb_win_config &cfg)