blob: ed308260e8ba0236ead195e60d8c12392f79ee15 [file] [log] [blame]
// Copyright 2019 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "cast/streaming/answer_messages.h"
#include <utility>
#include "absl/strings/str_cat.h"
#include "absl/strings/str_split.h"
#include "platform/base/error.h"
#include "util/json/json_helpers.h"
#include "util/osp_logging.h"
namespace openscreen {
namespace cast {
namespace {
/// Constraint properties.
// Audio constraints. See properties below.
static constexpr char kAudio[] = "audio";
// Video constraints. See properties below.
static constexpr char kVideo[] = "video";
// An optional field representing the minimum bits per second. If not specified
// by the receiver, the sender will use kDefaultAudioMinBitRate and
// kDefaultVideoMinBitRate, which represent the true operational minimum.
static constexpr char kMinBitRate[] = "minBitRate";
// 32kbps is sender default for audio minimum bit rate.
static constexpr int kDefaultAudioMinBitRate = 32 * 1000;
// 300kbps is sender default for video minimum bit rate.
static constexpr int kDefaultVideoMinBitRate = 300 * 1000;
// Maximum encoded bits per second. This is the lower of (1) the max capability
// of the decoder, or (2) the max data transfer rate.
static constexpr char kMaxBitRate[] = "maxBitRate";
// Maximum supported end-to-end latency, in milliseconds. Proportional to the
// size of the data buffers in the receiver.
static constexpr char kMaxDelay[] = "maxDelay";
/// Video constraint properties.
// Maximum pixel rate (width * height * framerate). Is often less than
// multiplying the fields in maxDimensions. This field is used to set the
// maximum processing rate.
static constexpr char kMaxPixelsPerSecond[] = "maxPixelsPerSecond";
// Minimum dimensions. If omitted, the sender will assume a reasonable minimum
// with the same aspect ratio as maxDimensions, as close to 320*180 as possible.
// Should reflect the true operational minimum.
static constexpr char kMinDimensions[] = "minDimensions";
// Maximum dimensions, not necessarily ideal dimensions.
static constexpr char kMaxDimensions[] = "maxDimensions";
/// Audio constraint properties.
// Maximum supported sampling frequency (not necessarily ideal).
static constexpr char kMaxSampleRate[] = "maxSampleRate";
// Maximum number of audio channels (1 is mono, 2 is stereo, etc.).
static constexpr char kMaxChannels[] = "maxChannels";
/// Dimension properties.
// Width in pixels.
static constexpr char kWidth[] = "width";
// Height in pixels.
static constexpr char kHeight[] = "height";
// Frame rate as a rational decimal number or fraction.
// E.g. 30 and "3000/1001" are both valid representations.
static constexpr char kFrameRate[] = "frameRate";
/// Display description properties
// If this optional field is included in the ANSWER message, the receiver is
// attached to a fixed display that has the given dimensions and frame rate
// configuration. These may exceed, be the same, or be less than the values in
// constraints. If undefined, we assume the display is not fixed (e.g. a Google
// Hangouts UI panel).
static constexpr char kDimensions[] = "dimensions";
// An optional field. When missing and dimensions are specified, the sender
// will assume square pixels and the dimensions imply the aspect ratio of the
// fixed display. WHen present and dimensions are also specified, implies the
// pixels are not square.
static constexpr char kAspectRatio[] = "aspectRatio";
// The delimeter used for the aspect ratio format ("A:B").
static constexpr char kAspectRatioDelimiter[] = ":";
// Sets the aspect ratio constraints. Value must be either "sender" or
// "receiver", see kScalingSender and kScalingReceiver below.
static constexpr char kScaling[] = "scaling";
// scaling = "sender" means that the sender must provide video frames of a fixed
// aspect ratio. In this case, the dimensions object must be passed or an error
// case will occur.
static constexpr char kScalingSender[] = "sender";
// scaling = "receiver" means that the sender may send arbitrarily sized frames,
// and the receiver will handle scaling and letterboxing as necessary.
static constexpr char kScalingReceiver[] = "receiver";
/// Answer properties.
// A number specifying the UDP port used for all streams in this session.
// Must have a value between kUdpPortMin and kUdpPortMax.
static constexpr char kUdpPort[] = "udpPort";
static constexpr int kUdpPortMin = 1;
static constexpr int kUdpPortMax = 65535;
// Numbers specifying the indexes chosen from the offer message.
static constexpr char kSendIndexes[] = "sendIndexes";
// uint32_t values specifying the RTP SSRC values used to send the RTCP feedback
// of the stream indicated in kSendIndexes.
static constexpr char kSsrcs[] = "ssrcs";
// Provides detailed maximum and minimum capabilities of the receiver for
// processing the selected streams. The sender may alter video resolution and
// frame rate throughout the session, and the constraints here determine how
// much data volume is allowed.
static constexpr char kConstraints[] = "constraints";
// Provides details about the display on the receiver.
static constexpr char kDisplay[] = "display";
// absl::optional array of numbers specifying the indexes of streams that will
// send event logs through RTCP.
static constexpr char kReceiverRtcpEventLog[] = "receiverRtcpEventLog";
// OPtional array of numbers specifying the indexes of streams that will use
// DSCP values specified in the OFFER message for RTCP packets.
static constexpr char kReceiverRtcpDscp[] = "receiverRtcpDscp";
// True if receiver can report wifi status.
static constexpr char kReceiverGetStatus[] = "receiverGetStatus";
// If this optional field is present the receiver supports the specific
// RTP extensions (such as adaptive playout delay).
static constexpr char kRtpExtensions[] = "rtpExtensions";
Json::Value AspectRatioConstraintToJson(AspectRatioConstraint aspect_ratio) {
switch (aspect_ratio) {
case AspectRatioConstraint::kVariable:
return Json::Value(kScalingReceiver);
case AspectRatioConstraint::kFixed:
default:
return Json::Value(kScalingSender);
}
}
bool AspectRatioConstraintParseAndValidate(const Json::Value& value,
AspectRatioConstraint* out) {
// the aspect ratio constraint is an optional field.
if (!value) {
return true;
}
std::string aspect_ratio;
if (!json::ParseAndValidateString(value, &aspect_ratio)) {
return false;
}
if (aspect_ratio == kScalingReceiver) {
*out = AspectRatioConstraint::kVariable;
return true;
} else if (aspect_ratio == kScalingSender) {
*out = AspectRatioConstraint::kFixed;
return true;
}
return false;
}
template <typename T>
Json::Value PrimitiveVectorToJson(const std::vector<T>& vec) {
Json::Value array(Json::ValueType::arrayValue);
array.resize(vec.size());
for (Json::Value::ArrayIndex i = 0; i < vec.size(); ++i) {
array[i] = Json::Value(vec[i]);
}
return array;
}
template <typename T>
bool ParseOptional(const Json::Value& value, absl::optional<T>* out) {
// It's fine if the value is empty.
if (!value) {
return true;
}
T tentative_out;
if (!T::ParseAndValidate(value, &tentative_out)) {
return false;
}
*out = tentative_out;
return true;
}
} // namespace
// static
bool AspectRatio::ParseAndValidate(const Json::Value& value, AspectRatio* out) {
std::string parsed_value;
if (!json::ParseAndValidateString(value, &parsed_value)) {
return false;
}
std::vector<absl::string_view> fields =
absl::StrSplit(parsed_value, kAspectRatioDelimiter);
if (fields.size() != 2) {
return false;
}
if (!absl::SimpleAtoi(fields[0], &out->width) ||
!absl::SimpleAtoi(fields[1], &out->height)) {
return false;
}
return out->IsValid();
}
bool AspectRatio::IsValid() const {
return width > 0 && height > 0;
}
// static
bool AudioConstraints::ParseAndValidate(const Json::Value& root,
AudioConstraints* out) {
if (!json::ParseAndValidateInt(root[kMaxSampleRate],
&(out->max_sample_rate)) ||
!json::ParseAndValidateInt(root[kMaxChannels], &(out->max_channels)) ||
!json::ParseAndValidateInt(root[kMaxBitRate], &(out->max_bit_rate)) ||
!json::ParseAndValidateMilliseconds(root[kMaxDelay], &(out->max_delay))) {
return false;
}
if (!json::ParseAndValidateInt(root[kMinBitRate], &(out->min_bit_rate))) {
out->min_bit_rate = kDefaultAudioMinBitRate;
}
return out->IsValid();
}
Json::Value AudioConstraints::ToJson() const {
OSP_DCHECK(IsValid());
Json::Value root;
root[kMaxSampleRate] = max_sample_rate;
root[kMaxChannels] = max_channels;
root[kMinBitRate] = min_bit_rate;
root[kMaxBitRate] = max_bit_rate;
root[kMaxDelay] = Json::Value::Int64(max_delay.count());
return root;
}
bool AudioConstraints::IsValid() const {
return max_sample_rate > 0 && max_channels > 0 && min_bit_rate > 0 &&
max_bit_rate >= min_bit_rate;
}
bool Dimensions::ParseAndValidate(const Json::Value& root, Dimensions* out) {
if (!json::ParseAndValidateInt(root[kWidth], &(out->width)) ||
!json::ParseAndValidateInt(root[kHeight], &(out->height)) ||
!json::ParseAndValidateSimpleFraction(root[kFrameRate],
&(out->frame_rate))) {
return false;
}
return out->IsValid();
}
bool Dimensions::IsValid() const {
return width > 0 && height > 0 && frame_rate.is_positive();
}
Json::Value Dimensions::ToJson() const {
OSP_DCHECK(IsValid());
Json::Value root;
root[kWidth] = width;
root[kHeight] = height;
root[kFrameRate] = frame_rate.ToString();
return root;
}
// static
bool VideoConstraints::ParseAndValidate(const Json::Value& root,
VideoConstraints* out) {
if (!json::ParseAndValidateDouble(root[kMaxPixelsPerSecond],
&(out->max_pixels_per_second)) ||
!Dimensions::ParseAndValidate(root[kMaxDimensions],
&(out->max_dimensions)) ||
!json::ParseAndValidateInt(root[kMaxBitRate], &(out->max_bit_rate)) ||
!json::ParseAndValidateMilliseconds(root[kMaxDelay], &(out->max_delay)) ||
!ParseOptional<Dimensions>(root[kMinDimensions],
&(out->min_dimensions))) {
return false;
}
if (!json::ParseAndValidateInt(root[kMinBitRate], &(out->min_bit_rate))) {
out->min_bit_rate = kDefaultVideoMinBitRate;
}
return out->IsValid();
}
bool VideoConstraints::IsValid() const {
return max_pixels_per_second > 0 && min_bit_rate > 0 &&
max_bit_rate > min_bit_rate && max_delay.count() > 0 &&
max_dimensions.IsValid() &&
(!min_dimensions.has_value() || min_dimensions->IsValid()) &&
max_dimensions.frame_rate.numerator > 0;
}
Json::Value VideoConstraints::ToJson() const {
OSP_DCHECK(IsValid());
Json::Value root;
root[kMaxPixelsPerSecond] = max_pixels_per_second;
if (min_dimensions.has_value()) {
root[kMinDimensions] = min_dimensions->ToJson();
}
root[kMaxDimensions] = max_dimensions.ToJson();
root[kMinBitRate] = min_bit_rate;
root[kMaxBitRate] = max_bit_rate;
root[kMaxDelay] = Json::Value::Int64(max_delay.count());
return root;
}
// static
bool Constraints::ParseAndValidate(const Json::Value& root, Constraints* out) {
if (!AudioConstraints::ParseAndValidate(root[kAudio], &(out->audio)) ||
!VideoConstraints::ParseAndValidate(root[kVideo], &(out->video))) {
return false;
}
return out->IsValid();
}
bool Constraints::IsValid() const {
return audio.IsValid() && video.IsValid();
}
Json::Value Constraints::ToJson() const {
OSP_DCHECK(IsValid());
Json::Value root;
root[kAudio] = audio.ToJson();
root[kVideo] = video.ToJson();
return root;
}
// static
bool DisplayDescription::ParseAndValidate(const Json::Value& root,
DisplayDescription* out) {
if (!ParseOptional<Dimensions>(root[kDimensions], &(out->dimensions)) ||
!ParseOptional<AspectRatio>(root[kAspectRatio], &(out->aspect_ratio))) {
return false;
}
AspectRatioConstraint constraint;
if (AspectRatioConstraintParseAndValidate(root[kScaling], &constraint)) {
out->aspect_ratio_constraint =
absl::optional<AspectRatioConstraint>(std::move(constraint));
} else {
out->aspect_ratio_constraint = absl::nullopt;
}
return out->IsValid();
}
bool DisplayDescription::IsValid() const {
// At least one of the properties must be set, and if a property is set
// it must be valid.
if (aspect_ratio.has_value() && !aspect_ratio->IsValid()) {
return false;
}
if (dimensions.has_value() && !dimensions->IsValid()) {
return false;
}
// Sender behavior is undefined if the aspect ratio is fixed but no
// dimensions or aspect ratio are provided.
if (aspect_ratio_constraint.has_value() &&
(aspect_ratio_constraint.value() == AspectRatioConstraint::kFixed) &&
!dimensions.has_value() && !aspect_ratio.has_value()) {
return false;
}
return aspect_ratio.has_value() || dimensions.has_value() ||
aspect_ratio_constraint.has_value();
}
Json::Value DisplayDescription::ToJson() const {
OSP_DCHECK(IsValid());
Json::Value root;
if (aspect_ratio.has_value()) {
root[kAspectRatio] = absl::StrCat(
aspect_ratio->width, kAspectRatioDelimiter, aspect_ratio->height);
}
if (dimensions.has_value()) {
root[kDimensions] = dimensions->ToJson();
}
if (aspect_ratio_constraint.has_value()) {
root[kScaling] =
AspectRatioConstraintToJson(aspect_ratio_constraint.value());
}
return root;
}
bool Answer::ParseAndValidate(const Json::Value& root, Answer* out) {
if (!json::ParseAndValidateInt(root[kUdpPort], &(out->udp_port)) ||
!json::ParseAndValidateIntArray(root[kSendIndexes],
&(out->send_indexes)) ||
!json::ParseAndValidateUintArray(root[kSsrcs], &(out->ssrcs)) ||
!ParseOptional<Constraints>(root[kConstraints], &(out->constraints)) ||
!ParseOptional<DisplayDescription>(root[kDisplay], &(out->display))) {
return false;
}
if (!json::ParseBool(root[kReceiverGetStatus],
&(out->supports_wifi_status_reporting))) {
out->supports_wifi_status_reporting = false;
}
// These function set to empty array if not present, so we can ignore
// the return value for optional values.
json::ParseAndValidateIntArray(root[kReceiverRtcpEventLog],
&(out->receiver_rtcp_event_log));
json::ParseAndValidateIntArray(root[kReceiverRtcpDscp],
&(out->receiver_rtcp_dscp));
json::ParseAndValidateStringArray(root[kRtpExtensions],
&(out->rtp_extensions));
return out->IsValid();
}
bool Answer::IsValid() const {
if (ssrcs.empty() || send_indexes.empty()) {
return false;
}
// We don't know what the indexes used in the offer were here, so we just
// sanity check.
for (const int index : send_indexes) {
if (index < 0) {
return false;
}
}
if (constraints.has_value() && !constraints->IsValid()) {
return false;
}
if (display.has_value() && !display->IsValid()) {
return false;
}
return kUdpPortMin <= udp_port && udp_port <= kUdpPortMax;
}
Json::Value Answer::ToJson() const {
OSP_DCHECK(IsValid());
Json::Value root;
if (constraints.has_value()) {
root[kConstraints] = constraints->ToJson();
}
if (display.has_value()) {
root[kDisplay] = display->ToJson();
}
root[kUdpPort] = udp_port;
root[kReceiverGetStatus] = supports_wifi_status_reporting;
root[kSendIndexes] = PrimitiveVectorToJson(send_indexes);
root[kSsrcs] = PrimitiveVectorToJson(ssrcs);
// Some sender do not handle empty array properly, so we omit these fields
// if they are empty.
if (!receiver_rtcp_event_log.empty()) {
root[kReceiverRtcpEventLog] =
PrimitiveVectorToJson(receiver_rtcp_event_log);
}
if (!receiver_rtcp_dscp.empty()) {
root[kReceiverRtcpDscp] = PrimitiveVectorToJson(receiver_rtcp_dscp);
}
if (!rtp_extensions.empty()) {
root[kRtpExtensions] = PrimitiveVectorToJson(rtp_extensions);
}
return root;
}
} // namespace cast
} // namespace openscreen