offeranswer.c 5.58 KiB
/*
linphone
Copyright (C) 2010  Simon MORLAT (simon.morlat@free.fr)
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, write to the Free Software
Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
#include "sal.h"
#include "offeranswer.h"
static PayloadType * find_payload_type_best_match(const MSList *l, const PayloadType *refpt){
	PayloadType *pt;
	char value[10];
	const MSList *elem;
	PayloadType *candidate=NULL;
	for (elem=l;elem!=NULL;elem=elem->next){
		pt=(PayloadType*)elem->data;
		/* the compare between G729 and G729A is for some stupid uncompliant phone*/
		if (strcasecmp(pt->mime_type,refpt->mime_type)==0  ||
		    ((strcasecmp(pt->mime_type, "G729") == 0 && strcasecmp(refpt->mime_type, "G729A") == 0 ))
			&& pt->clock_rate==refpt->clock_rate){
			candidate=pt;
			/*good candidate, check fmtp for H264 */
			if (strcasecmp(pt->mime_type,"H264")==0){
				if (pt->recv_fmtp!=NULL && refpt->recv_fmtp!=NULL){
					int mode1=0,mode2=0;
					if (fmtp_get_value(pt->recv_fmtp,"packetization-mode",value,sizeof(value))){
						mode1=atoi(value);
					if (fmtp_get_value(refpt->recv_fmtp,"packetization-mode",value,sizeof(value))){
						mode2=atoi(value);
					if (mode1==mode2)
					    break; /*exact match */
			}else break;
	return candidate;
static MSList *match_payloads(const MSList *local, const MSList *remote){
	const MSList *e2;
	MSList *res=NULL;
	PayloadType *matched;
	for(e2=remote;e2!=NULL;e2=e2->next){
		PayloadType *p2=(PayloadType*)e2->data;
		matched=find_payload_type_best_match(local,p2);
		if (matched){
			matched=payload_type_clone(matched);
			if (p2->send_fmtp)
				payload_type_set_send_fmtp(matched,p2->send_fmtp);
			res=ms_list_append(res,matched);
			payload_type_set_number(matched,payload_type_get_number(p2));
		}else{
			ms_message("No match for %s/%i",p2->mime_type,p2->clock_rate);
7172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140
} } return res; } static bool_t only_telephone_event(const MSList *l){ PayloadType *p=(PayloadType*)l->data; if (strcasecmp(p->mime_type,"telephone-event")!=0){ return FALSE; } return TRUE; } static void initiate_outgoing(const SalStreamDescription *local_offer, const SalStreamDescription *remote_answer, SalStreamDescription *result){ if (remote_answer->port!=0) result->payloads=match_payloads(local_offer->payloads,remote_answer->payloads); result->proto=local_offer->proto; result->type=local_offer->type; result->dir=local_offer->dir; if (result->payloads && !only_telephone_event(result->payloads)){ strcpy(result->addr,remote_answer->addr); result->port=remote_answer->port; result->bandwidth=remote_answer->bandwidth; result->ptime=remote_answer->ptime; }else{ result->port=0; } } static void initiate_incoming(const SalStreamDescription *local_cap, const SalStreamDescription *remote_offer, SalStreamDescription *result){ result->payloads=match_payloads(local_cap->payloads,remote_offer->payloads); result->proto=local_cap->proto; result->type=local_cap->type; if (remote_offer->dir==SalStreamSendOnly) result->dir=SalStreamRecvOnly; else if (remote_offer->dir==SalStreamRecvOnly){ result->dir=SalStreamSendOnly; }else if (remote_offer->dir==SalStreamInactive){ result->dir=SalStreamInactive; }else result->dir=SalStreamSendRecv; if (result->payloads && !only_telephone_event(result->payloads)){ strcpy(result->addr,local_cap->addr); result->port=local_cap->port; result->bandwidth=local_cap->bandwidth; result->ptime=local_cap->ptime; }else{ result->port=0; } } /** * Returns a media description to run the streams with, based on a local offer * and the returned response (remote). **/ int offer_answer_initiate_outgoing(const SalMediaDescription *local_offer, const SalMediaDescription *remote_answer, SalMediaDescription *result){ int i,j; const SalStreamDescription *ls,*rs; for(i=0,j=0;i<local_offer->nstreams;++i){ ms_message("Processing for stream %i",i); ls=&local_offer->streams[i]; rs=sal_media_description_find_stream(remote_answer,ls->proto,ls->type); if (rs) {
141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177
initiate_outgoing(ls,rs,&result->streams[j]); ++j; } else ms_warning("No matching stream for %i",i); } result->nstreams=j; strcpy(result->addr,remote_answer->addr); return 0; } /** * Returns a media description to run the streams with, based on the local capabilities and * and the received offer. * The returned media description is an answer and should be sent to the offerer. **/ int offer_answer_initiate_incoming(const SalMediaDescription *local_capabilities, const SalMediaDescription *remote_offer, SalMediaDescription *result){ int i,j; const SalStreamDescription *ls,*rs; for(i=0,j=0;i<remote_offer->nstreams;++i){ rs=&remote_offer->streams[i]; ms_message("Processing for stream %i",i); ls=sal_media_description_find_stream(local_capabilities,rs->proto,rs->type); if (ls){ initiate_incoming(ls,rs,&result->streams[j]); ++j; } } result->nstreams=j; strcpy(result->username, local_capabilities->username); strcpy(result->addr,local_capabilities->addr); return 0; }