mirror of
https://github.com/Atmosphere-NX/Atmosphere-libs.git
synced 2025-06-21 11:02:45 +02:00
150 lines
7.1 KiB
C++
150 lines
7.1 KiB
C++
/*
|
|
* Copyright (c) Atmosphère-NX
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
* under the terms and conditions of the GNU General Public License,
|
|
* version 2, as published by the Free Software Foundation.
|
|
*
|
|
* This program is distributed in the hope it will be useful, but WITHOUT
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
* more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
#include <stratosphere.hpp>
|
|
#include "decodersrv_decoder_server_object.hpp"
|
|
#include "../jpeg/decodersrv_software_jpeg_decoder.hpp"
|
|
#include "../jpeg/decodersrv_software_jpeg_shrinker.hpp"
|
|
|
|
namespace ams::capsrv::server {
|
|
|
|
namespace {
|
|
|
|
constexpr const int JpegShrinkQualities[] = {
|
|
98, 95, 90, 80, 70, 60, 50, 40, 30, 20, 10, 0
|
|
};
|
|
|
|
Result DecodeJpegImpl(void *dst, size_t dst_size, const void *src_jpeg, size_t src_jpeg_size, u32 width, u32 height, const ScreenShotDecodeOption &option, void *work, size_t work_size) {
|
|
/* Clear the work memory. */
|
|
std::memset(work, 0, work_size);
|
|
ON_SCOPE_EXIT { std::memset(work, 0, work_size); };
|
|
|
|
/* Clear the output memory. */
|
|
std::memset(dst, 0, dst_size);
|
|
auto clear_guard = SCOPE_GUARD { std::memset(dst, 0, dst_size); };
|
|
|
|
/* Validate parameters. */
|
|
R_UNLESS(util::IsAligned(width, 0x10), capsrv::ResultAlbumOutOfRange());
|
|
R_UNLESS(util::IsAligned(height, 0x4), capsrv::ResultAlbumOutOfRange());
|
|
|
|
R_UNLESS(dst != nullptr, capsrv::ResultAlbumReadBufferShortage());
|
|
R_UNLESS(dst_size >= 4 * width * height, capsrv::ResultAlbumReadBufferShortage());
|
|
|
|
R_UNLESS(src_jpeg != nullptr, capsrv::ResultAlbumInvalidFileData());
|
|
R_UNLESS(src_jpeg_size != 0, capsrv::ResultAlbumInvalidFileData());
|
|
|
|
/* Create the input. */
|
|
const jpeg::SoftwareJpegDecoderInput decode_input = {
|
|
.jpeg = src_jpeg,
|
|
.jpeg_size = src_jpeg_size,
|
|
.width = width,
|
|
.height = height,
|
|
.fancy_upsampling = option.HasJpegDecoderFlag(ScreenShotDecoderFlag_EnableFancyUpsampling),
|
|
.block_smoothing = option.HasJpegDecoderFlag(ScreenShotDecoderFlag_EnableBlockSmoothing),
|
|
};
|
|
|
|
/* Create the output. */
|
|
s32 out_width = 0, out_height = 0;
|
|
jpeg::SoftwareJpegDecoderOutput decode_output = {
|
|
.out_width = std::addressof(out_width),
|
|
.out_height = std::addressof(out_height),
|
|
.dst = dst,
|
|
.dst_size = dst_size,
|
|
};
|
|
|
|
/* Decode the jpeg. */
|
|
R_TRY(jpeg::SoftwareJpegDecoder::DecodeRgba8(decode_output, decode_input, work, work_size));
|
|
|
|
/* We succeeded, so we shouldn't clear the output memory. */
|
|
clear_guard.Cancel();
|
|
R_SUCCEED();
|
|
}
|
|
|
|
Result ShrinkJpegImpl(u64 *out_size, void *dst, size_t dst_size, const void *src_jpeg, size_t src_jpeg_size, u32 width, u32 height, const ScreenShotDecodeOption &option, void *work, size_t work_size) {
|
|
/* Validate parameters. */
|
|
R_UNLESS(util::IsAligned(width, 0x10), capsrv::ResultAlbumOutOfRange());
|
|
R_UNLESS(util::IsAligned(height, 0x4), capsrv::ResultAlbumOutOfRange());
|
|
|
|
R_UNLESS(dst != nullptr, capsrv::ResultInternalJpegOutBufferShortage());
|
|
R_UNLESS(dst_size != 0, capsrv::ResultAlbumReadBufferShortage());
|
|
|
|
R_UNLESS(src_jpeg != nullptr, capsrv::ResultAlbumInvalidFileData());
|
|
R_UNLESS(src_jpeg_size != 0, capsrv::ResultAlbumInvalidFileData());
|
|
|
|
/* Create the input. */
|
|
const jpeg::SoftwareJpegShrinkerInput shrink_input = {
|
|
.jpeg = src_jpeg,
|
|
.jpeg_size = src_jpeg_size,
|
|
.width = width,
|
|
.height = height,
|
|
.fancy_upsampling = option.HasJpegDecoderFlag(ScreenShotDecoderFlag_EnableFancyUpsampling),
|
|
.block_smoothing = option.HasJpegDecoderFlag(ScreenShotDecoderFlag_EnableBlockSmoothing),
|
|
};
|
|
|
|
/* Create the output. */
|
|
u64 shrunk_size = 0;
|
|
s32 shrunk_width = 0, shrunk_height = 0;
|
|
jpeg::SoftwareJpegShrinkerOutput shrink_output = {
|
|
.out_size = std::addressof(shrunk_size),
|
|
.out_width = std::addressof(shrunk_width),
|
|
.out_height = std::addressof(shrunk_height),
|
|
.dst = dst,
|
|
.dst_size = dst_size,
|
|
};
|
|
|
|
/* Try to shrink the jpeg at various quality levels. */
|
|
for (auto quality : JpegShrinkQualities) {
|
|
/* Shrink at the current quality. */
|
|
R_TRY_CATCH(jpeg::SoftwareJpegShrinker::ShrinkRgba8(shrink_output, shrink_input, quality, work, work_size)) {
|
|
/* If the output buffer isn't large enough to fit the output, we should try at a lower quality. */
|
|
R_CATCH(capsrv::ResultInternalJpegOutBufferShortage) {
|
|
continue;
|
|
}
|
|
/* Nintendo doesn't catch this result, but our lack of work buffer use makes me think this may be necessary. */
|
|
R_CATCH(capsrv::ResultInternalJpegWorkMemoryShortage) {
|
|
continue;
|
|
}
|
|
} R_END_TRY_CATCH;
|
|
|
|
/* Write the output size. */
|
|
*out_size = shrunk_size;
|
|
R_SUCCEED();
|
|
}
|
|
|
|
/* Nintendo aborts if no quality succeeds. */
|
|
AMS_ABORT("ShrinkJpeg should succeed before this point\n");
|
|
}
|
|
|
|
}
|
|
|
|
Result DecoderControlService::DecodeJpeg(const ams::sf::OutNonSecureBuffer &out, const ams::sf::InBuffer &in, u32 width, u32 height, const ScreenShotDecodeOption &option) {
|
|
/* Get the work buffer. */
|
|
void *work = g_work_memory.jpeg_decoder_memory;
|
|
size_t work_size = sizeof(g_work_memory.jpeg_decoder_memory);
|
|
|
|
/* Call the decoder implementation. */
|
|
R_RETURN(DecodeJpegImpl(out.GetPointer(), out.GetSize(), in.GetPointer(), in.GetSize(), width, height, option, work, work_size));
|
|
}
|
|
|
|
Result DecoderControlService::ShrinkJpeg(ams::sf::Out<u64> out_size, const ams::sf::OutNonSecureBuffer &out, const ams::sf::InBuffer &in, u32 width, u32 height, const capsrv::ScreenShotDecodeOption &option) {
|
|
/* Get the work buffer. */
|
|
void *work = g_work_memory.jpeg_decoder_memory;
|
|
size_t work_size = sizeof(g_work_memory.jpeg_decoder_memory);
|
|
|
|
/* Call the shrink implementation. */
|
|
R_RETURN(ShrinkJpegImpl(out_size.GetPointer(), out.GetPointer(), out.GetSize(), in.GetPointer(), in.GetSize(), width, height, option, work, work_size));
|
|
}
|
|
|
|
} |