You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
sems/apps/sbc/SDPFilter.cpp

247 lines
7.9 KiB

/*
* Copyright (C) 2010 Stefan Sayer
*
* This file is part of SEMS, a free SIP media server.
*
* SEMS 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.
*
* For a license to use the SEMS software under conditions
* other than those described here, or to purchase support for this
* software, please contact iptel.org by e-mail at the following addresses:
* info@iptel.org
*
* SEMS 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, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
#include "SDPFilter.h"
#include <algorithm>
#include "log.h"
#include "AmUtils.h"
#include "RTPParameters.h"
int filterSDP(AmSdp& sdp, const vector<FilterEntry>& filter_list) {
for (vector<FilterEntry>::const_iterator it=
filter_list.begin(); it!=filter_list.end(); it++){
const FilterType& sdpfilter = it->filter_type;
const std::set<string>& sdpfilter_list = it->filter_list;
bool media_line_filtered_out = false;
bool media_line_left = false;
if (!isActiveFilter(sdpfilter))
continue;
for (std::vector<SdpMedia>::iterator m_it =
sdp.media.begin(); m_it != sdp.media.end(); m_it++) {
SdpMedia& media = *m_it;
std::vector<SdpPayload> new_pl;
for (std::vector<SdpPayload>::iterator p_it =
media.payloads.begin(); p_it != media.payloads.end(); p_it++) {
string c = p_it->encoding_name;
std::transform(c.begin(), c.end(), c.begin(), ::tolower);
bool is_filtered = (sdpfilter == Whitelist) ^
(sdpfilter_list.find(c) != sdpfilter_list.end());
// DBG("%s (%s) is_filtered: %s\n", p_it->encoding_name.c_str(), c.c_str(),
// is_filtered?"true":"false");
if (!is_filtered)
new_pl.push_back(*p_it);
}
// if no payload supported any more: leave at least one, and set port to 0
// RFC 3264 section 6.1
if(!new_pl.size() && media.payloads.size()) {
std::vector<SdpPayload>::iterator p_it = media.payloads.begin();
new_pl.push_back(*p_it);
media.port = 0;
media_line_filtered_out = true;
}
else media_line_left = true;
media.payloads = new_pl;
}
if ((!media_line_left) && media_line_filtered_out) {
// no filter adds new payloads, we can safely return error
DBG("all streams were marked as inactive\n");
return -488;
}
}
return 0;
}
int filterMedia(AmSdp& sdp, const vector<FilterEntry>& filter_list)
{
unsigned filtered_out = 0;
DBG("filtering media types\n");
for (vector<FilterEntry>::const_iterator i = filter_list.begin(); i !=filter_list.end(); ++i) {
const FilterType& filter = i->filter_type;
const std::set<string>& media_list = i->filter_list;
if (!isActiveFilter(filter)) continue;
for (std::vector<SdpMedia>::iterator m = sdp.media.begin(); m != sdp.media.end(); ++m) {
if (m->port == 0) continue; // already inactive
string type(SdpMedia::type2str(m->type));
DBG("checking whether to filter out '%s'\n", type.c_str());
bool is_filtered = (filter == Whitelist) ^ (media_list.find(type) != media_list.end());
if (is_filtered) {
m->port = 0;
filtered_out++;
}
}
}
if (filtered_out > 0 && filtered_out == sdp.media.size()) {
// we filtered out all media lines (if there was an inactive stream before
// it was probably intendend)
DBG("all streams were marked as inactive\n");
return -488;
}
return 0;
}
void fix_missing_encodings(SdpMedia& m) {
for (std::vector<SdpPayload>::iterator p_it=
m.payloads.begin(); p_it!=m.payloads.end(); p_it++) {
SdpPayload& p = *p_it;
if (!p.encoding_name.empty())
continue;
if (p.payload_type > (IANA_RTP_PAYLOADS_SIZE-1) || p.payload_type < 0)
continue; // todo: throw out this payload
if (IANA_RTP_PAYLOADS[p.payload_type].payload_name[0]=='\0')
continue; // todo: throw out this payload
p.encoding_name = IANA_RTP_PAYLOADS[p.payload_type].payload_name;
p.clock_rate = IANA_RTP_PAYLOADS[p.payload_type].clock_rate;
if (IANA_RTP_PAYLOADS[p.payload_type].channels > 1)
p.encoding_param = IANA_RTP_PAYLOADS[p.payload_type].channels;
DBG("named SDP payload type %d with %s/%d%s\n",
p.payload_type, IANA_RTP_PAYLOADS[p.payload_type].payload_name,
IANA_RTP_PAYLOADS[p.payload_type].clock_rate,
IANA_RTP_PAYLOADS[p.payload_type].channels > 1 ?
("/"+int2str(IANA_RTP_PAYLOADS[p.payload_type].channels)).c_str() : "");
}
}
void fix_incomplete_silencesupp(SdpMedia& m) {
for (std::vector<SdpAttribute>::iterator a_it =
m.attributes.begin(); a_it != m.attributes.end(); a_it++) {
if (a_it->attribute == "silenceSupp") {
vector<string> parts = explode(a_it->value, " ");
if (parts.size() < 5) {
string val_before = a_it->value;
for (int i=parts.size();i<5;i++)
a_it->value += " -";
DBG("fixed SDP attribute silenceSupp:'%s' -> '%s'\n",
val_before.c_str(), a_it->value.c_str());
}
}
}
}
std::vector<SdpAttribute> filterSDPAttributes(std::vector<SdpAttribute> attributes,
FilterType sdpalinesfilter, const std::set<string>& sdpalinesfilter_list) {
std::vector<SdpAttribute> res;
for (std::vector<SdpAttribute>::iterator a_it =
attributes.begin(); a_it != attributes.end(); a_it++) {
// Case insensitive search:
string c = a_it->attribute;
std::transform(c.begin(), c.end(), c.begin(), ::tolower);
// Check, if this should be filtered:
bool is_filtered = (sdpalinesfilter == Whitelist) ^
(sdpalinesfilter_list.find(c) != sdpalinesfilter_list.end());
DBG("%s (%s) is_filtered: %s\n", a_it->attribute.c_str(), c.c_str(),
is_filtered?"true":"false");
// If it is not filtered, just add it to the list:
if (!is_filtered)
res.push_back(*a_it);
}
return res;
}
int filterSDPalines(AmSdp& sdp, const vector<FilterEntry>& filter_list) {
for (vector<FilterEntry>::const_iterator it=
filter_list.begin(); it!=filter_list.end(); it++){
const FilterType& sdpalinesfilter = it->filter_type;
const std::set<string>& sdpalinesfilter_list = it->filter_list;
// If not Black- or Whitelist, simply return
if (!isActiveFilter(sdpalinesfilter))
continue;
// We start with per Session-alines
sdp.attributes =
filterSDPAttributes(sdp.attributes, sdpalinesfilter, sdpalinesfilter_list);
for (std::vector<SdpMedia>::iterator m_it =
sdp.media.begin(); m_it != sdp.media.end(); m_it++) {
SdpMedia& media = *m_it;
// todo: what if no payload supported any more?
media.attributes =
filterSDPAttributes(media.attributes, sdpalinesfilter, sdpalinesfilter_list);
}
}
return 0;
}
int normalizeSDP(AmSdp& sdp, bool anonymize_sdp, const string &advertised_ip) {
for (std::vector<SdpMedia>::iterator m_it=
sdp.media.begin(); m_it != sdp.media.end(); m_it++) {
if (m_it->type != MT_AUDIO && m_it->type != MT_VIDEO)
continue;
// fill missing encoding names (a= lines)
fix_missing_encodings(*m_it);
// fix incomplete silenceSupp attributes (see RFC3108)
// (only media level - RFC3108 4.)
fix_incomplete_silencesupp(*m_it);
}
if (anonymize_sdp) {
// Clear s-Line in SDP:
sdp.sessionName = "-";
// Clear u-Line in SDP:
sdp.uri.clear();
// Clear origin user
sdp.origin.user = "-";
if (!advertised_ip.empty()) {
sdp.origin.conn.address = advertised_ip;
// SdpConnection::ipv4/ipv6 seems not to be used so we won't replace it there
// TODO: replace in attributes
}
}
return 0;
}