The spec for nal_ref_idc doesn't indicate that it should be used for priority like x264 seems to be using it for. NVENC seems to pass different values, so let's not rely on it. The new scheme might be naive, but we can iterate on it, and apply evenly to all H.264 encoders.
234 lines
5.8 KiB
C
234 lines
5.8 KiB
C
/******************************************************************************
|
|
Copyright (C) 2014 by Hugh Bailey <obs.jim@gmail.com>
|
|
|
|
This program 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 2 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program 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 this program. If not, see <http://www.gnu.org/licenses/>.
|
|
******************************************************************************/
|
|
|
|
#include "obs-avc.h"
|
|
|
|
#include "obs.h"
|
|
#include "obs-nal.h"
|
|
#include "util/array-serializer.h"
|
|
|
|
bool obs_avc_keyframe(const uint8_t *data, size_t size)
|
|
{
|
|
const uint8_t *nal_start, *nal_end;
|
|
const uint8_t *end = data + size;
|
|
|
|
nal_start = obs_nal_find_startcode(data, end);
|
|
while (true) {
|
|
while (nal_start < end && !*(nal_start++))
|
|
;
|
|
|
|
if (nal_start == end)
|
|
break;
|
|
|
|
const uint8_t type = nal_start[0] & 0x1F;
|
|
|
|
if (type == OBS_NAL_SLICE_IDR || type == OBS_NAL_SLICE)
|
|
return type == OBS_NAL_SLICE_IDR;
|
|
|
|
nal_end = obs_nal_find_startcode(nal_start, end);
|
|
nal_start = nal_end;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
const uint8_t *obs_avc_find_startcode(const uint8_t *p, const uint8_t *end)
|
|
{
|
|
return obs_nal_find_startcode(p, end);
|
|
}
|
|
|
|
static void serialize_avc_data(struct serializer *s, const uint8_t *data,
|
|
size_t size, bool *is_keyframe, int *priority)
|
|
{
|
|
const uint8_t *const end = data + size;
|
|
const uint8_t *nal_start = obs_nal_find_startcode(data, end);
|
|
while (true) {
|
|
while (nal_start < end && !*(nal_start++))
|
|
;
|
|
|
|
if (nal_start == end)
|
|
break;
|
|
|
|
const int type = nal_start[0] & 0x1F;
|
|
|
|
switch (type) {
|
|
case OBS_NAL_SLICE:
|
|
if (*priority < OBS_NAL_PRIORITY_HIGH)
|
|
*priority = OBS_NAL_PRIORITY_HIGH;
|
|
break;
|
|
case OBS_NAL_SLICE_IDR:
|
|
*is_keyframe = true;
|
|
*priority = OBS_NAL_PRIORITY_HIGHEST;
|
|
}
|
|
|
|
const uint8_t *const nal_end =
|
|
obs_nal_find_startcode(nal_start, end);
|
|
const size_t size = nal_end - nal_start;
|
|
s_wb32(s, (uint32_t)size);
|
|
s_write(s, nal_start, size);
|
|
nal_start = nal_end;
|
|
}
|
|
}
|
|
|
|
void obs_parse_avc_packet(struct encoder_packet *avc_packet,
|
|
const struct encoder_packet *src)
|
|
{
|
|
struct array_output_data output;
|
|
struct serializer s;
|
|
long ref = 1;
|
|
|
|
array_output_serializer_init(&s, &output);
|
|
*avc_packet = *src;
|
|
|
|
serialize(&s, &ref, sizeof(ref));
|
|
serialize_avc_data(&s, src->data, src->size, &avc_packet->keyframe,
|
|
&avc_packet->priority);
|
|
|
|
avc_packet->data = output.bytes.array + sizeof(ref);
|
|
avc_packet->size = output.bytes.num - sizeof(ref);
|
|
avc_packet->drop_priority = avc_packet->priority;
|
|
}
|
|
|
|
static inline bool has_start_code(const uint8_t *data)
|
|
{
|
|
if (data[0] != 0 || data[1] != 0)
|
|
return false;
|
|
|
|
return data[2] == 1 || (data[2] == 0 && data[3] == 1);
|
|
}
|
|
|
|
static void get_sps_pps(const uint8_t *data, size_t size, const uint8_t **sps,
|
|
size_t *sps_size, const uint8_t **pps, size_t *pps_size)
|
|
{
|
|
const uint8_t *nal_start, *nal_end;
|
|
const uint8_t *end = data + size;
|
|
int type;
|
|
|
|
nal_start = obs_nal_find_startcode(data, end);
|
|
while (true) {
|
|
while (nal_start < end && !*(nal_start++))
|
|
;
|
|
|
|
if (nal_start == end)
|
|
break;
|
|
|
|
nal_end = obs_nal_find_startcode(nal_start, end);
|
|
|
|
type = nal_start[0] & 0x1F;
|
|
if (type == OBS_NAL_SPS) {
|
|
*sps = nal_start;
|
|
*sps_size = nal_end - nal_start;
|
|
} else if (type == OBS_NAL_PPS) {
|
|
*pps = nal_start;
|
|
*pps_size = nal_end - nal_start;
|
|
}
|
|
|
|
nal_start = nal_end;
|
|
}
|
|
}
|
|
|
|
size_t obs_parse_avc_header(uint8_t **header, const uint8_t *data, size_t size)
|
|
{
|
|
struct array_output_data output;
|
|
struct serializer s;
|
|
const uint8_t *sps = NULL, *pps = NULL;
|
|
size_t sps_size = 0, pps_size = 0;
|
|
|
|
array_output_serializer_init(&s, &output);
|
|
|
|
if (size <= 6)
|
|
return 0;
|
|
|
|
if (!has_start_code(data)) {
|
|
*header = bmemdup(data, size);
|
|
return size;
|
|
}
|
|
|
|
get_sps_pps(data, size, &sps, &sps_size, &pps, &pps_size);
|
|
if (!sps || !pps || sps_size < 4)
|
|
return 0;
|
|
|
|
s_w8(&s, 0x01);
|
|
s_write(&s, sps + 1, 3);
|
|
s_w8(&s, 0xff);
|
|
s_w8(&s, 0xe1);
|
|
|
|
s_wb16(&s, (uint16_t)sps_size);
|
|
s_write(&s, sps, sps_size);
|
|
s_w8(&s, 0x01);
|
|
s_wb16(&s, (uint16_t)pps_size);
|
|
s_write(&s, pps, pps_size);
|
|
|
|
*header = output.bytes.array;
|
|
return output.bytes.num;
|
|
}
|
|
|
|
void obs_extract_avc_headers(const uint8_t *packet, size_t size,
|
|
uint8_t **new_packet_data, size_t *new_packet_size,
|
|
uint8_t **header_data, size_t *header_size,
|
|
uint8_t **sei_data, size_t *sei_size)
|
|
{
|
|
DARRAY(uint8_t) new_packet;
|
|
DARRAY(uint8_t) header;
|
|
DARRAY(uint8_t) sei;
|
|
const uint8_t *nal_start, *nal_end, *nal_codestart;
|
|
const uint8_t *end = packet + size;
|
|
|
|
da_init(new_packet);
|
|
da_init(header);
|
|
da_init(sei);
|
|
|
|
nal_start = obs_nal_find_startcode(packet, end);
|
|
nal_end = NULL;
|
|
while (nal_end != end) {
|
|
nal_codestart = nal_start;
|
|
|
|
while (nal_start < end && !*(nal_start++))
|
|
;
|
|
|
|
if (nal_start == end)
|
|
break;
|
|
|
|
const uint8_t type = nal_start[0] & 0x1F;
|
|
|
|
nal_end = obs_nal_find_startcode(nal_start, end);
|
|
if (!nal_end)
|
|
nal_end = end;
|
|
|
|
if (type == OBS_NAL_SPS || type == OBS_NAL_PPS) {
|
|
da_push_back_array(header, nal_codestart,
|
|
nal_end - nal_codestart);
|
|
} else if (type == OBS_NAL_SEI) {
|
|
da_push_back_array(sei, nal_codestart,
|
|
nal_end - nal_codestart);
|
|
|
|
} else {
|
|
da_push_back_array(new_packet, nal_codestart,
|
|
nal_end - nal_codestart);
|
|
}
|
|
|
|
nal_start = nal_end;
|
|
}
|
|
|
|
*new_packet_data = new_packet.array;
|
|
*new_packet_size = new_packet.num;
|
|
*header_data = header.array;
|
|
*header_size = header.num;
|
|
*sei_data = sei.array;
|
|
*sei_size = sei.num;
|
|
}
|