/* ** ** Copyright 2008, The Android Open Source Project ** Copyright 2010, Samsung Electronics Co. LTD ** ** Licensed under the Apache License, Version 2.0 (the "License"); ** you may not use this file except in compliance with the License. ** You may obtain a copy of the License at ** ** http://www.apache.org/licenses/LICENSE-2.0 ** ** Unless required by applicable law or agreed to in writing, software ** distributed under the License is distributed on an "AS IS" BASIS, ** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. ** See the License for the specific language governing permissions and ** limitations under the License. */ //#define LOG_NDEBUG 0 #define LOG_TAG "Jpeg-api" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #ifdef S5P_VMEM #include "s5p_vmem_api.h" #endif #include "jpeg_api.h" static struct jpeg_lib *jpeg_ctx = NULL; #ifdef S5P_VMEM static int mem_fp; unsigned int cookie; #endif /* S5P_VMEM */ static unsigned int get_yuv_size(enum jpeg_frame_format out_format, unsigned int width, unsigned int height) { switch (out_format) { case YUV_422 : if (width % 16 != 0) width += 16 - (width % 16); if (height % 8 != 0) height += 8 - (height % 8); break; case YUV_420 : if (width % 16 != 0) width += 16 - (width % 16); if (height % 16 != 0) height += 16 - (height % 16); break; default: ALOGV("get_yuv_size return fmt(%d)\n", out_format); return(0); } ALOGV("get_yuv_size width(%d) height(%d)\n", width, height); switch (out_format) { case YUV_422 : return(width*height*2); case YUV_420 : return((width*height) + (width*height >> 1)); default : return(0); } } static int check_input_size(unsigned int width, unsigned int height) { if ((width % 16) != 0 || (height % 8) != 0) return -1; return 0; } static void init_decode_param(void) { jpeg_ctx = (struct jpeg_lib *)malloc(sizeof(struct jpeg_lib)); memset(jpeg_ctx, 0x00, sizeof(struct jpeg_lib)); jpeg_ctx->args.dec_param = (struct jpeg_dec_param *)malloc(sizeof(struct jpeg_dec_param)); memset(jpeg_ctx->args.dec_param, 0x00, sizeof(struct jpeg_dec_param)); } static void init_encode_param(void) { jpeg_ctx = (struct jpeg_lib *)malloc(sizeof(struct jpeg_lib)); memset(jpeg_ctx, 0x00, sizeof(struct jpeg_lib)); jpeg_ctx->args.enc_param = (struct jpeg_enc_param *)malloc(sizeof(struct jpeg_enc_param)); memset(jpeg_ctx->args.enc_param, 0x00, sizeof(struct jpeg_enc_param)); } int api_jpeg_decode_init() { init_decode_param(); jpeg_ctx->jpeg_fd = open(JPEG_DRIVER_NAME, O_RDWR); if (jpeg_ctx->jpeg_fd < 0) { ALOGE("JPEG driver open failed\n"); return -1; } #ifdef S5P_VMEM mem_fp = s5p_vmem_open(); ALOGV("s5p_vmem_open\n"); #else jpeg_ctx->args.mmapped_addr = (char *) mmap(0, JPEG_TOTAL_BUF_SIZE, PROT_READ | PROT_WRITE, MAP_SHARED, jpeg_ctx->jpeg_fd, 0); if (jpeg_ctx->args.mmapped_addr == NULL) { ALOGE("JPEG mmap failed\n"); return -1; } ALOGV("api_jpeg_decode_init jpeg_ctx->args.mmapped_addr 0x%08x\n", jpeg_ctx->args.mmapped_addr); #endif /* S5P_VMEM */ return jpeg_ctx->jpeg_fd; } int api_jpeg_encode_init() { init_encode_param(); jpeg_ctx->jpeg_fd = open(JPEG_DRIVER_NAME, O_RDWR); if (jpeg_ctx->jpeg_fd < 0) { ALOGE("JPEG driver open failed %d\n", jpeg_ctx->jpeg_fd); return -1; } #ifdef S5P_VMEM mem_fp = s5p_vmem_open(); ALOGI("s5p_vmem_open\n"); #else jpeg_ctx->args.mmapped_addr = (char *) mmap(0, JPEG_TOTAL_BUF_SIZE, PROT_READ | PROT_WRITE, MAP_SHARED, jpeg_ctx->jpeg_fd, 0); if (jpeg_ctx->args.mmapped_addr == NULL) { ALOGE("JPEG mmap failed\n"); return -1; } ALOGV("api_jpeg_encode_init jpeg_ctx->args.mmapped_addr 0x%08x\n", jpeg_ctx->args.mmapped_addr); #endif /* S5P_VMEM */ return jpeg_ctx->jpeg_fd; } int api_jpeg_decode_deinit(int dev_fd) { if (jpeg_ctx->args.mmapped_addr != NULL) munmap(jpeg_ctx->args.mmapped_addr, JPEG_TOTAL_BUF_SIZE); #ifdef S5P_VMEM s5p_free_share(mem_fp, jpeg_ctx->args.in_cookie, jpeg_ctx->args.in_buf); s5p_free_share(mem_fp, jpeg_ctx->args.out_cookie, jpeg_ctx->args.out_buf); s5p_vmem_close(mem_fp); #endif close(jpeg_ctx->jpeg_fd); if (jpeg_ctx->args.dec_param != NULL) free(jpeg_ctx->args.dec_param); free(jpeg_ctx); return JPEG_OK; } int api_jpeg_encode_deinit(int dev_fd) { if (jpeg_ctx->args.mmapped_addr != NULL) munmap(jpeg_ctx->args.mmapped_addr, JPEG_TOTAL_BUF_SIZE); #ifdef S5P_VMEM s5p_free_share(mem_fp, jpeg_ctx->args.in_cookie, jpeg_ctx->args.in_buf); s5p_free_share(mem_fp, jpeg_ctx->args.out_cookie, jpeg_ctx->args.out_buf); s5p_vmem_close(mem_fp); #endif close(jpeg_ctx->jpeg_fd); if (jpeg_ctx->args.enc_param != NULL) free(jpeg_ctx->args.enc_param); free(jpeg_ctx); return JPEG_OK; } void *api_jpeg_get_decode_in_buf(int dev_fd, unsigned int size) { if (size < 0 || size > MAX_JPEG_RES) { ALOGE("Invalid decode input buffer size\r\n"); return NULL; } #ifdef S5P_VMEM jpeg_ctx->args.in_cookie = (unsigned int)ioctl(jpeg_ctx->jpeg_fd, IOCTL_GET_DEC_IN_BUF, size); jpeg_ctx->args.in_buf = s5p_malloc_share(mem_fp, jpeg_ctx->args.in_cookie, &jpeg_ctx->args.in_buf_size); #else jpeg_ctx->args.in_buf = (char *)ioctl(jpeg_ctx->jpeg_fd, IOCTL_GET_DEC_IN_BUF, jpeg_ctx->args.mmapped_addr); #endif /* S5P_VMEM */ return (void *)(jpeg_ctx->args.in_buf); } void *api_jpeg_get_encode_in_buf(int dev_fd, unsigned int size) { #ifdef S5P_VMEM jpeg_ctx->args.in_cookie = (unsigned int)ioctl(jpeg_ctx->jpeg_fd, IOCTL_GET_ENC_IN_BUF, (size*3)); jpeg_ctx->args.in_buf = s5p_malloc_share(mem_fp, jpeg_ctx->args.in_cookie, &jpeg_ctx->args.in_buf_size); #else jpeg_ctx->args.enc_param->size = size; jpeg_ctx->args.in_buf = (char *)ioctl(jpeg_ctx->jpeg_fd, IOCTL_GET_ENC_IN_BUF, jpeg_ctx->args.mmapped_addr); #endif ALOGV("api_jpeg_get_encode_in_buf: 0x%x\n", jpeg_ctx->args.in_buf); return (void *)(jpeg_ctx->args.in_buf); } void *api_jpeg_get_decode_out_buf(int dev_fd) { #ifdef S5P_VMEM jpeg_ctx->args.out_cookie = (unsigned int)ioctl(jpeg_ctx->jpeg_fd, IOCTL_GET_DEC_OUT_BUF, JPEG_FRAME_BUF_SIZE); jpeg_ctx->args.out_buf = s5p_malloc_share(mem_fp, jpeg_ctx->args.out_cookie, &jpeg_ctx->args.out_buf_size); #else jpeg_ctx->args.out_buf = (char *)ioctl(jpeg_ctx->jpeg_fd, IOCTL_GET_DEC_OUT_BUF, jpeg_ctx->args.mmapped_addr); #endif /* S5P_VMEM */ /* ALOGV("api_jpeg_get_decode_out_buf: 0x%x\n", jpeg_ctx->args.out_buf); */ return (void *)(jpeg_ctx->args.out_buf); } void *api_jpeg_get_encode_out_buf(int dev_fd) { #ifdef S5P_VMEM jpeg_ctx->args.out_cookie = (unsigned int)ioctl(jpeg_ctx->jpeg_fd, IOCTL_GET_ENC_OUT_BUF, JPEG_STREAM_BUF_SIZE); jpeg_ctx->args.out_buf = s5p_malloc_share(mem_fp, jpeg_ctx->args.out_cookie, &jpeg_ctx->args.out_buf_size); #else jpeg_ctx->args.out_buf = (char *)ioctl(jpeg_ctx->jpeg_fd, IOCTL_GET_ENC_OUT_BUF, jpeg_ctx->args.mmapped_addr); #endif /* S5P_VMEM */ ALOGV("api_jpeg_get_encode_out_buf: 0x%x\n", jpeg_ctx->args.out_buf); return (void *)(jpeg_ctx->args.out_buf); } void api_jpeg_set_decode_param(struct jpeg_dec_param *param) { memcpy(jpeg_ctx->args.dec_param, param, sizeof(struct jpeg_dec_param)); ioctl(jpeg_ctx->jpeg_fd, IOCTL_SET_DEC_PARAM, jpeg_ctx->args.dec_param); } void api_jpeg_set_encode_param(struct jpeg_enc_param *param) { memcpy(jpeg_ctx->args.enc_param, param, sizeof(struct jpeg_enc_param)); ioctl(jpeg_ctx->jpeg_fd, IOCTL_SET_ENC_PARAM, jpeg_ctx->args.enc_param); } enum jpeg_ret_type api_jpeg_decode_exe(int dev_fd, struct jpeg_dec_param *dec_param) { struct jpeg_args *arg; arg = &(jpeg_ctx->args); ioctl(jpeg_ctx->jpeg_fd, IOCTL_JPEG_DEC_EXE, arg->dec_param); ALOGV("api_jpeg_decode_exe dec_param->out_fmt :%d \ dec_param->width : %d dec_param->height : %d\n", arg->dec_param->out_fmt, arg->dec_param->width, arg->dec_param->height); dec_param->width = arg->dec_param->width; dec_param->height = arg->dec_param->height; dec_param->size = get_yuv_size(arg->dec_param->out_fmt, arg->dec_param->width, arg->dec_param->height); return JPEG_DECODE_OK; } enum jpeg_ret_type api_jpeg_encode_exe(int dev_fd, struct jpeg_enc_param *enc_param) { struct jpeg_args *arg; arg = &(jpeg_ctx->args); // check MCU validation width & height & sampling mode if (check_input_size(jpeg_ctx->args.enc_param->width, jpeg_ctx->args.enc_param->height) < 0) { ALOGV("width/height doesn't match with MCU\r\n"); return JPEG_FAIL; } ioctl(jpeg_ctx->jpeg_fd, IOCTL_JPEG_ENC_EXE, arg->enc_param); enc_param->size = arg->enc_param->size; return JPEG_ENCODE_OK; }