Files
UltraGrid/src/video.cpp
2020-04-29 09:34:09 +02:00

237 lines
7.2 KiB
C++

/**
* @file video.cpp
* @author Martin Pulec <pulec@cesnet.cz>
*
* @brief This file defines some common video functions.
*
* These function are neither video frame nor video codec related.
*/
/*
* Copyright (c) 2013 CESNET, z. s. p. o.
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, is permitted provided that the following conditions
* are met:
*
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
*
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* 3. Neither the name of CESNET nor the names of its contributors may be
* used to endorse or promote products derived from this software without
* specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHORS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESSED OR IMPLIED WARRANTIES, INCLUDING,
* BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
* AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO
* EVENT SHALL THE AUTHORS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
* INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
* OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#ifdef HAVE_CONFIG_H
#include "config.h"
#include "config_unix.h"
#include "config_win32.h"
#endif // HAVE_CONFIG_H
#include "debug.h"
#include "video.h"
#include <iomanip>
#include <sstream>
#include <map>
using namespace std;
typedef struct {
const char *name;
int x;
int y;
} video_mode_info_t;
const static map<enum video_mode, video_mode_info_t> video_mode_info = {
{ VIDEO_UNKNOWN, { "(unknown)", 0, 0 }},
{ VIDEO_NORMAL, { "normal", 1, 1 }},
{ VIDEO_DUAL, { "dual-link", 1, 2 }},
{ VIDEO_STEREO, { "3D", 2, 1 }},
{ VIDEO_4K, { "tiled-2x2", 2, 2 }},
{ VIDEO_3X1, { "3x1", 3, 1 }},
};
/**
* This function matches string representation of video mode with its
* respective enumeration value.
*
* @param reuqested_mode textual representation of video mode
* @return valid member of enum video mode
*/
enum video_mode get_video_mode_from_str(const char *requested_mode) {
if(strcasecmp(requested_mode, "help") == 0) {
printf("Video mode options:\n");
auto it = video_mode_info.begin();
while (it != video_mode_info.end()) {
if (it == video_mode_info.begin()) { // omit unknown
++it;
continue;
}
printf("\t%s\n", it->second.name);
if (++it != video_mode_info.end()) {
}
}
return VIDEO_UNKNOWN;
} else {
for (auto it : video_mode_info) {
if (strcasecmp(requested_mode, it.second.name) == 0) {
return it.first;
}
}
fprintf(stderr, "Unknown video mode (see -M help)\n");
return VIDEO_UNKNOWN;
}
}
int get_video_mode_tiles_x(enum video_mode video_mode)
{
return video_mode_info.at(video_mode).x;
}
int get_video_mode_tiles_y(enum video_mode video_mode)
{
return video_mode_info.at(video_mode).y;
}
const char *get_video_mode_description(enum video_mode video_mode)
{
return video_mode_info.at(video_mode).name;
}
enum video_mode guess_video_mode(int num_substreams)
{
assert(num_substreams > 0);
switch (num_substreams) {
case 1:
return VIDEO_NORMAL;
case 2:
return VIDEO_STEREO;
case 3:
return VIDEO_3X1;
case 4:
return VIDEO_4K;
default:
return VIDEO_UNKNOWN;
}
}
std::ostream& operator<<(std::ostream& os, const video_desc& desc)
{
std::streamsize p = os.precision();
ios_base::fmtflags f = os.flags();
if (desc.tile_count > 1) {
os << desc.tile_count << "*";
}
os << desc.width << "x" << desc.height << " @" << setprecision(2) << setiosflags(ios_base::fixed)
<< desc.fps * (desc.interlacing == PROGRESSIVE || desc.interlacing == SEGMENTED_FRAME ? 1 : 2)
<< get_interlacing_suffix(desc.interlacing) << ", codec "
<< get_codec_name(desc.color_spec);
os.precision(p);
os.flags(f);
return os;
}
std::ostream& operator<<(std::ostream& os, const codec_t& color_spec)
{
os << get_codec_name(color_spec);
return os;
}
std::istream& operator>>(std::istream& is, video_desc& desc)
{
video_desc out;
char buf[1024];
char *line = buf;
char interl_suffix[4] = "";
char codec_name[11] = "";
out.tile_count = 1;
unsigned int i = 0;
int num_spaces = 0;
while (is.good() && i < sizeof buf - 1) {
char c;
is.get(c);
if (!is.good()) {
break;
}
if (num_spaces == 3 &&
!(isalnum(c) || c == '.')) // allowed symbols in codec name
{
is.unget();
break;
} else {
buf[i++] = c;
}
if (c == ' ') {
num_spaces += 1;
}
}
buf[i] = '\0';
/// @todo regex matching would be better
if (strchr(line, '*')) {
out.tile_count = atoi(line);
line = strchr(line, '*') + 1;
}
int ret = sscanf(line, "%dx%d @%lf%3[a-z], codec %10s", &out.width, &out.height,
&out.fps, interl_suffix, codec_name);
if (ret != 5) {
is.setstate(std::ios::failbit);
log_msg(LOG_LEVEL_ERROR, "Malformed video_desc representation!\n");
return is;
}
out.color_spec = get_codec_from_name(codec_name);
out.interlacing = get_interlacing_from_suffix(interl_suffix);
out.fps = out.fps / (out.interlacing == PROGRESSIVE || out.interlacing == SEGMENTED_FRAME ? 1 : 2);
desc = out;
return is;
}
bool video_desc::operator==(video_desc const & other) const
{
return video_desc_eq(*this, other);
}
bool video_desc::operator!=(video_desc const & other) const
{
return !video_desc_eq(*this, other);
}
bool video_desc::operator!() const
{
return color_spec == VIDEO_CODEC_NONE;
}
video_desc::operator string() const
{
ostringstream oss;
oss << *this;
return oss.str();
}