view sip-in/prack.c @ 109:9b87894704eb

sip-in: first step toward final call clearing
author Mychaela Falconia <falcon@freecalypso.org>
date Wed, 28 Sep 2022 16:32:13 -0800
parents 0d6435808bcd
children c1c94b7fc2e2
line wrap: on
line source

/*
 * Here we implement our handling of SIP PRACK, expected from callers
 * when we send them a reliable 180 Ringing response.
 */

#include <sys/types.h>
#include <sys/socket.h>
#include <sys/time.h>
#include <netinet/in.h>
#include <ctype.h>
#include <stdio.h>
#include <stdint.h>
#include <stdlib.h>
#include <string.h>
#include <strings.h>
#include <syslog.h>
#include "../include/gsm48_const.h"
#include "../libsip/parse.h"
#include "../libsip/uas_basic.h"
#include "../libsip/out_msg.h"
#include "call.h"

extern char *get_single_header();
extern struct call *find_call_by_sip_id();

extern unsigned sip_linger_error;

void
handle_sip_prack(req, ess, sin)
	struct sip_pkt_rx *req;
	struct uas_parse_hdrs *ess;
	struct sockaddr_in *sin;
{
	struct call *call;
	struct sip_msg_out resp;
	char *rack, *orig_method, *cp;
	unsigned rseq, orig_num;
	int rc;

	call = find_call_by_sip_id(ess->call_id);
	if (!call) {
		start_response_out_msg(&resp, "481 Call-ID not found");
error_resp:	rc = add_resp_basic_headers(&resp, ess, req->req_method);
		if (rc < 0)
			return;
		out_msg_finish(&resp);
		sip_tx_packet(&resp, sin);
		return;
	}
	rack = get_single_header(req, "RAck", (char *) 0, (int *) 0);
	if (!rack) {
		start_response_out_msg(&resp, "400 Missing RAck header");
		goto error_resp;
	}
	if (!isdigit(*rack)) {
malformed:	start_response_out_msg(&resp, "400 Malformed RAck header");
		goto error_resp;
	}
	rseq = strtoul(rack, &cp, 10);
	if (!isspace(*cp))
		goto malformed;
	while (isspace(*cp))
		cp++;
	if (!isdigit(*cp))
		goto malformed;
	orig_num = strtoul(cp, &cp, 10);
	if (!isspace(*cp))
		goto malformed;
	while (isspace(*cp))
		cp++;
	if (!isupper(*cp))
		goto malformed;
	orig_method = cp;
	while (isalnum(*cp))
		cp++;
	if (*cp)
		goto malformed;
	if (rseq != 1 || orig_num != call->invite_cseq ||
	    strcmp(orig_method, "INVITE")) {
		start_response_out_msg(&resp,
			"481 RAck fails to match our 100rel response");
		goto error_resp;
	}
	switch (call->sip_state) {
	case SIP_STATE_RINGING_REL:
		call->sip_state = SIP_STATE_RINGING;
		start_response_out_msg(&resp, "200 OK");
		rc = add_resp_basic_headers(&resp, ess, req->req_method);
		if (rc < 0) {
			syslog(LOG_ERR, "PRACK 200 response length exceeded");
			call->sip_state = SIP_STATE_MSG_SIZE_ERR;
			call->overall_state = OVERALL_STATE_TEARDOWN;
			disconnect_mncc(call, GSM48_CAUSE_LOC_PRN_S_LU,
					GSM48_CC_CAUSE_INTERWORKING);
			disconnect_tmgw(call);
			sip_mark_end_time(call, sip_linger_error);
			/* TODO: transition from TEARDOWN to DEAD_SIP */
			return;
		}
		out_msg_finish(&resp);
		sip_tx_packet(&resp, sin);
		return;
	case SIP_STATE_RINGING:
	case SIP_STATE_INVITE_200:
		start_response_out_msg(&resp, "200 OK");
		rc = add_resp_basic_headers(&resp, ess, req->req_method);
		if (rc < 0)
			return;
		out_msg_finish(&resp);
		sip_tx_packet(&resp, sin);
		return;
	case SIP_STATE_INVITE_PROC:
	case SIP_STATE_CONNECTED:
	case SIP_STATE_BYE_SENT:
	case SIP_STATE_INVITE_ERR:
	case SIP_STATE_ENDED:
		start_response_out_msg(&resp,
				       "481 No outstanding 100rel response");
		goto error_resp;
	}
}