| /* |
| ** |
| ** 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 <utils/Log.h> |
| #include <stdio.h> |
| #include <stdlib.h> |
| #include <sys/types.h> |
| #include <sys/stat.h> |
| #include <sys/ioctl.h> |
| #include <fcntl.h> |
| #include <ctype.h> |
| #include <unistd.h> |
| #include <sys/mman.h> |
| #include <string.h> |
| #include <errno.h> |
| #include <sys/time.h> |
| #include <signal.h> |
| #include <math.h> |
| #include <sys/poll.h> |
| |
| #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; |
| } |