You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
230 lines
5.5 KiB
230 lines
5.5 KiB
/*
|
|
* Copyright Samsung Electronics Co.,LTD.
|
|
* Copyright (C) 2015 The Android Open Source Project
|
|
*
|
|
* 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.
|
|
*/
|
|
|
|
#include <linux/videodev2.h>
|
|
|
|
#include <ExynosJpegApi.h>
|
|
|
|
#include "hwjpeg-internal.h"
|
|
|
|
int ExynosJpegEncoder::setJpegConfig(void* pConfig)
|
|
{
|
|
ExynosJpegEncoder *that = reinterpret_cast<ExynosJpegEncoder *>(pConfig);
|
|
|
|
if (!setColorFormat(that->m_v4l2Format))
|
|
return -1;
|
|
|
|
if (!setJpegFormat(that->m_jpegFormat))
|
|
return -1;
|
|
|
|
if (!setSize(that->m_nWidth, that->m_nHeight))
|
|
return -1;
|
|
|
|
m_iInBufType = that->m_iInBufType;
|
|
m_iOutBufType = that->m_iOutBufType;
|
|
|
|
return 0;
|
|
}
|
|
|
|
int ExynosJpegEncoder::getInBuf(int *piBuf, int *piInputSize, int iSize)
|
|
{
|
|
if (iSize < 1) {
|
|
ALOGE("Invalid array size %d for getInBuf()", iSize);
|
|
return -1;
|
|
}
|
|
|
|
size_t len_buffers[iSize];
|
|
if (!m_hwjpeg.GetImageBuffers(piBuf, len_buffers, static_cast<unsigned int>(iSize)))
|
|
return -1;
|
|
|
|
for (int i = 0; i < iSize; i++)
|
|
piInputSize[i] = static_cast<int>(len_buffers[i]);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int ExynosJpegEncoder::getOutBuf(int *piBuf, int *piOutputSize)
|
|
{
|
|
size_t len;
|
|
if (!m_hwjpeg.GetJpegBuffer(piBuf, &len))
|
|
return -1;
|
|
|
|
*piOutputSize = static_cast<int>(len);
|
|
return 0;
|
|
}
|
|
|
|
int ExynosJpegEncoder::setInBuf(int *piBuf, int *iSize)
|
|
{
|
|
size_t buflen[3];
|
|
unsigned int bufnum = 3;
|
|
|
|
if (!EnsureFormatIsApplied())
|
|
return -1;
|
|
|
|
if (!m_hwjpeg.GetImageBufferSizes(buflen, &bufnum))
|
|
return -1;
|
|
|
|
for (unsigned int i = 0; i < bufnum; i++)
|
|
buflen[i] = static_cast<size_t>(iSize[i]);
|
|
|
|
if (!m_hwjpeg.SetImageBuffer(piBuf, buflen, bufnum))
|
|
return -1;
|
|
|
|
m_iInBufType = JPEG_BUF_TYPE_DMA_BUF;
|
|
|
|
return 0;
|
|
}
|
|
|
|
int ExynosJpegEncoder::setOutBuf(int iBuf, int iSize, int offset)
|
|
{
|
|
if (!m_hwjpeg.SetJpegBuffer(iBuf, static_cast<size_t>(iSize), offset))
|
|
return -1;
|
|
|
|
m_iOutBufType = JPEG_BUF_TYPE_DMA_BUF;
|
|
|
|
return 0;
|
|
}
|
|
|
|
int ExynosJpegEncoder::getInBuf(char **pcBuf, int *piInputSize, int iSize)
|
|
{
|
|
if (iSize < 1) {
|
|
ALOGE("Invalid array size %d for getInBuf()", iSize);
|
|
return -1;
|
|
}
|
|
|
|
size_t len_buffers[iSize];
|
|
if (!m_hwjpeg.GetImageBuffers(pcBuf, len_buffers, static_cast<unsigned int>(iSize)))
|
|
return -1;
|
|
|
|
for (int i = 0; i < iSize; i++)
|
|
piInputSize[i] = static_cast<int>(len_buffers[i]);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int ExynosJpegEncoder::getOutBuf(char **pcBuf, int *piOutputSize)
|
|
{
|
|
size_t len;
|
|
if (!m_hwjpeg.GetJpegBuffer(pcBuf, &len))
|
|
return -1;
|
|
|
|
*piOutputSize = static_cast<int>(len);
|
|
return 0;
|
|
}
|
|
|
|
int ExynosJpegEncoder::setInBuf(char **pcBuf, int *iSize)
|
|
{
|
|
size_t buflen[3];
|
|
unsigned int bufnum = 3;
|
|
|
|
if (!EnsureFormatIsApplied())
|
|
return -1;
|
|
|
|
if (!m_hwjpeg.GetImageBufferSizes(buflen, &bufnum))
|
|
return -1;
|
|
|
|
for (unsigned int i = 0; i < bufnum; i++)
|
|
buflen[i] = static_cast<size_t>(iSize[i]);
|
|
|
|
if (!m_hwjpeg.SetImageBuffer(pcBuf, buflen, bufnum))
|
|
return -1;
|
|
|
|
m_iInBufType = JPEG_BUF_TYPE_USER_PTR;
|
|
return 0;
|
|
}
|
|
|
|
int ExynosJpegEncoder::setOutBuf(char *pcBuf, int iSize)
|
|
{
|
|
if (!m_hwjpeg.SetJpegBuffer(pcBuf, static_cast<size_t>(iSize)))
|
|
return -1;
|
|
|
|
m_iOutBufType = JPEG_BUF_TYPE_USER_PTR;
|
|
|
|
return 0;
|
|
}
|
|
|
|
int ExynosJpegEncoder::setJpegFormat(int iV4l2JpegFormat)
|
|
{
|
|
if (m_jpegFormat == iV4l2JpegFormat)
|
|
return 0;
|
|
|
|
unsigned int hfactor, vfactor;
|
|
switch (iV4l2JpegFormat) {
|
|
case V4L2_PIX_FMT_JPEG_444:
|
|
hfactor = 1;
|
|
vfactor = 1;
|
|
break;
|
|
case V4L2_PIX_FMT_JPEG_422:
|
|
hfactor = 2;
|
|
vfactor = 1;
|
|
break;
|
|
case V4L2_PIX_FMT_JPEG_420:
|
|
hfactor = 2;
|
|
vfactor = 2;
|
|
break;
|
|
case V4L2_PIX_FMT_JPEG_GRAY:
|
|
hfactor = 0;
|
|
vfactor = 0;
|
|
break;
|
|
case V4L2_PIX_FMT_JPEG_422V:
|
|
hfactor = 1;
|
|
vfactor = 2;
|
|
break;
|
|
case V4L2_PIX_FMT_JPEG_411:
|
|
hfactor = 4;
|
|
vfactor = 1;
|
|
break;
|
|
default:
|
|
ALOGE("Unknown JPEG format `%08Xh", iV4l2JpegFormat);
|
|
return -1;
|
|
}
|
|
|
|
if (!m_hwjpeg.SetChromaSampFactor(hfactor, vfactor))
|
|
return -1;
|
|
|
|
m_jpegFormat = iV4l2JpegFormat;
|
|
|
|
return 0;
|
|
}
|
|
|
|
int ExynosJpegEncoder::setColorBufSize(int *piBufSize, int iSize)
|
|
{
|
|
size_t len[3];
|
|
unsigned int num = static_cast<unsigned int>(iSize);
|
|
|
|
if (!m_hwjpeg.GetImageBufferSizes(len, &num))
|
|
return -1;
|
|
|
|
for (unsigned int i = 0; i < num; i++)
|
|
piBufSize[i] = static_cast<int>(len[i]);
|
|
|
|
return 0;
|
|
}
|
|
|
|
bool ExynosJpegEncoder::__EnsureFormatIsApplied() {
|
|
if (TestStateEither(STATE_SIZE_CHANGED | STATE_PIXFMT_CHANGED) &&
|
|
!m_hwjpeg.SetImageFormat(m_v4l2Format, m_nWidth, m_nHeight))
|
|
return false;
|
|
|
|
ClearState(STATE_SIZE_CHANGED | STATE_PIXFMT_CHANGED);
|
|
return true;
|
|
}
|
|
|
|
int ExynosJpegEncoder::setQuality(const unsigned char q_table[]) {
|
|
return m_hwjpeg.SetQuality(q_table) ? 0 : -1;
|
|
}
|