/* * This file is part of Moonlight Embedded. * * Copyright (C) 2015 Iwan Timmer * * Moonlight is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 3 of the License, or * (at your option) any later version. * * Moonlight is distributed in the hope that 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 Moonlight; if not, see . */ #include "sps.h" #include "h264_stream.h" static h264_stream_t* h264_stream; static int initial_width, initial_height; void gs_sps_init(int width, int height) { h264_stream = h264_new(); initial_width = width; initial_height = height; } void gs_sps_fix(PLENTRY sps, int flags, uint8_t* out_buf, uint32_t* out_offset) { int start_len = sps->data[2] == 0x01 ? 3 : 4; read_nal_unit(h264_stream, sps->data+start_len, sps->length-start_len); // Some decoders rely on H264 level to decide how many buffers are needed // Since we only need one frame buffered, we'll set level as low as we can // for known resolution combinations. Otherwise leave the profile alone (currently 5.0) if (initial_width == 1280 && initial_height == 720) h264_stream->sps->level_idc = 32; // Max 5 buffered frames at 1280x720x60 else if (initial_width == 1920 && initial_height == 1080) h264_stream->sps->level_idc = 42; // Max 4 buffered frames at 1920x1080x60 // Some decoders requires a reference frame count of 1 to decode successfully. h264_stream->sps->num_ref_frames = 1; // GFE 2.5.11 changed the SPS to add additional extensions // Some devices don't like these so we remove them here. if (flags & GS_SPS_REMOVE_VST_FIXUP) h264_stream->sps->vui.video_signal_type_present_flag = 0; if (flags & GS_SPS_REMOVE_CLI_FIXUP) h264_stream->sps->vui.chroma_loc_info_present_flag = 0; if ((flags & GS_SPS_BITSTREAM_FIXUP) == GS_SPS_BITSTREAM_FIXUP) { // The SPS that comes in the current H264 bytestream doesn't set the bitstream_restriction_flag // or the max_dec_frame_buffering which increases decoding latency on some devices. // log2_max_mv_length_horizontal and log2_max_mv_length_vertical are set to more // conservative values by GFE 2.5.11. We'll let those values stand. if (!h264_stream->sps->vui.bitstream_restriction_flag) { h264_stream->sps->vui.bitstream_restriction_flag = 1; h264_stream->sps->vui.motion_vectors_over_pic_boundaries_flag = 1; h264_stream->sps->vui.max_bits_per_mb_denom = 1; h264_stream->sps->vui.log2_max_mv_length_horizontal = 16; h264_stream->sps->vui.log2_max_mv_length_vertical = 16; h264_stream->sps->vui.num_reorder_frames = 0; } // Some devices throw errors if max_dec_frame_buffering < num_ref_frames h264_stream->sps->vui.max_dec_frame_buffering = 1; // These values are the default for the fields, but they are more aggressive // than what GFE sends in 2.5.11, but it doesn't seem to cause picture problems. h264_stream->sps->vui.max_bytes_per_pic_denom = 2; h264_stream->sps->vui.max_bits_per_mb_denom = 1; } memcpy(out_buf+*out_offset, sps->data, start_len); *out_offset += start_len; *out_offset += write_nal_unit(h264_stream, out_buf+*out_offset, 128); }