aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authormatteo <matteo@f38db490-d61c-443f-a65b-d21fe96a405b>2003-03-03 06:00:17 +0000
committermatteo <matteo@f38db490-d61c-443f-a65b-d21fe96a405b>2003-03-03 06:00:17 +0000
commitddf7eb769521e76b87e0400c3bee66372111259c (patch)
tree8947999511995c4b3abb14d82903f896602886da
parent48085489fc4adfc22d5d3e568ac68abc25bd0f31 (diff)
Mon Mar 3 07:00:01 CET 2003
git-svn-id: http://svn.digium.com/svn/asterisk/trunk@630 f38db490-d61c-443f-a65b-d21fe96a405b
-rwxr-xr-xchannels/chan_iax2.c58
1 files changed, 46 insertions, 12 deletions
diff --git a/channels/chan_iax2.c b/channels/chan_iax2.c
index 76efd99e5..03cc83393 100755
--- a/channels/chan_iax2.c
+++ b/channels/chan_iax2.c
@@ -349,6 +349,8 @@ struct ast_iax2_frame {
struct ast_frame *f;
/* /Our/ call number */
unsigned short callno;
+ /* /Their/ call number */
+ unsigned short dcallno;
/* Start of raw frame (outgoing only) */
void *data;
/* Length of frame (outgoing only) */
@@ -566,8 +568,11 @@ void showframe(struct ast_iax2_frame *f, struct ast_iax2_full_hdr *fhi, int rx,
fh = f->data;
snprintf(retries, sizeof(retries), "%03d", f->retries);
} else {
- strcpy(retries, "N/A");
fh = fhi;
+ if (ntohs(fh->dcallno) & AST_FLAG_RETRANS)
+ strcpy(retries, "Yes");
+ else
+ strcpy(retries, "No");
}
if (!(ntohs(fh->scallno) & AST_FLAG_FULL)) {
/* Don't mess with mini-frames */
@@ -1147,7 +1152,17 @@ static void iax2_destroy_nolock(int callno)
ast_pthread_mutex_lock(&iaxsl[callno]);
}
-
+static int update_packet(struct ast_iax2_frame *f)
+{
+ /* Called with iaxsl lock held, and iaxs[callno] non-NULL */
+ struct ast_iax2_full_hdr *fh = f->data;
+ /* Mark this as a retransmission */
+ fh->dcallno = ntohs(AST_FLAG_RETRANS | f->dcallno);
+ /* Update iseqno */
+ f->iseqno = iaxs[f->callno]->iseqno;
+ fh->iseqno = ntohs(f->iseqno);
+ return 0;
+}
static int attempt_transmit(void *data)
{
@@ -1193,6 +1208,8 @@ static int attempt_transmit(void *data)
}
freeme++;
} else {
+ /* Update it if it needs it */
+ update_packet(f);
/* Attempt transmission */
send_packet(f);
f->retries++;
@@ -2221,9 +2238,10 @@ static int iax2_send(struct chan_iax2_pvt *pvt, struct ast_frame *f, unsigned in
fh->type = fr->f->frametype & 0xFF;
fh->csub = compress_subclass(fr->f->subclass);
if (transfer) {
- fh->dcallno = htons(pvt->transfercallno);
+ fr->dcallno = pvt->transfercallno;
} else
- fh->dcallno = htons(pvt->peercallno);
+ fr->dcallno = pvt->peercallno;
+ fh->dcallno = htons(fr->dcallno);
fr->datalen = fr->f->datalen + sizeof(struct ast_iax2_full_hdr);
fr->data = fh;
fr->retries = 0;
@@ -3610,6 +3628,14 @@ static int socket_read(int *id, int fd, short events, void *cbdata)
(f.frametype != AST_FRAME_IAX))
iaxs[fr.callno]->iseqno++;
}
+ /* A full frame */
+ if (res < sizeof(struct ast_iax2_full_hdr)) {
+ ast_log(LOG_WARNING, "midget packet received (%d of %d min)\n", res, sizeof(struct ast_iax2_full_hdr));
+ ast_pthread_mutex_unlock(&iaxsl[fr.callno]);
+ return 1;
+ }
+ f.datalen = res - sizeof(struct ast_iax2_full_hdr);
+
/* Handle implicit ACKing unless this is an INVAL */
if (((f.subclass != AST_IAX2_COMMAND_INVAL)) ||
(f.frametype != AST_FRAME_IAX)) {
@@ -3643,17 +3669,17 @@ static int socket_read(int *id, int fd, short events, void *cbdata)
ast_pthread_mutex_unlock(&iaxq.lock);
}
/* Note how much we've received acknowledgement for */
- iaxs[fr.callno]->rseqno = fr.iseqno;
+ if (iaxs[fr.callno])
+ iaxs[fr.callno]->rseqno = fr.iseqno;
+ else {
+ /* Stop processing now */
+ ast_pthread_mutex_unlock(&iaxsl[fr.callno]);
+ return 1;
+ }
} else
ast_log(LOG_DEBUG, "Received iseqno %d not within window %d->%d\n", fr.iseqno, iaxs[fr.callno]->rseqno, iaxs[fr.callno]->oseqno);
}
- /* A full frame */
- if (res < sizeof(struct ast_iax2_full_hdr)) {
- ast_log(LOG_WARNING, "midget packet received (%d of %d min)\n", res, sizeof(struct ast_iax2_full_hdr));
- ast_pthread_mutex_unlock(&iaxsl[fr.callno]);
- return 1;
- }
- f.datalen = res - sizeof(struct ast_iax2_full_hdr);
+
if (f.datalen) {
if (f.frametype == AST_FRAME_IAX) {
if (parse_ies(&ies, buf + sizeof(struct ast_iax2_full_hdr), f.datalen)) {
@@ -3826,6 +3852,8 @@ static int socket_read(int *id, int fd, short events, void *cbdata)
case AST_IAX2_COMMAND_HANGUP:
iaxs[fr.callno]->alreadygone = 1;
ast_log(LOG_DEBUG, "Immediately destroying %d, having received hangup\n", fr.callno);
+ /* Send ack immediately, before we destroy */
+ send_command_immediate(iaxs[fr.callno], AST_FRAME_IAX, AST_IAX2_COMMAND_ACK, fr.ts, NULL, 0,fr.iseqno);
iax2_destroy_nolock(fr.callno);
break;
case AST_IAX2_COMMAND_REJECT:
@@ -3833,6 +3861,8 @@ static int socket_read(int *id, int fd, short events, void *cbdata)
ast_log(LOG_WARNING, "Call rejected by %s: %s\n", inet_ntoa(iaxs[fr.callno]->addr.sin_addr), ies.cause ? ies.cause : "<Unknown>");
iaxs[fr.callno]->error = EPERM;
ast_log(LOG_DEBUG, "Immediately destroying %d, having received reject\n", fr.callno);
+ /* Send ack immediately, before we destroy */
+ send_command_immediate(iaxs[fr.callno], AST_FRAME_IAX, AST_IAX2_COMMAND_ACK, fr.ts, NULL, 0,fr.iseqno);
iax2_destroy_nolock(fr.callno);
break;
case AST_IAX2_COMMAND_ACCEPT:
@@ -4074,6 +4104,8 @@ static int socket_read(int *id, int fd, short events, void *cbdata)
case AST_IAX2_COMMAND_REGACK:
if (iax2_ack_registry(&ies, &sin, fr.callno))
ast_log(LOG_WARNING, "Registration failure\n");
+ /* Send ack immediately, before we destroy */
+ send_command_immediate(iaxs[fr.callno], AST_FRAME_IAX, AST_IAX2_COMMAND_ACK, fr.ts, NULL, 0,fr.iseqno);
iax2_destroy_nolock(fr.callno);
break;
case AST_IAX2_COMMAND_REGREJ:
@@ -4081,6 +4113,8 @@ static int socket_read(int *id, int fd, short events, void *cbdata)
ast_log(LOG_NOTICE, "Registration of '%s' rejected: %s\n", iaxs[fr.callno]->reg->username, ies.cause ? ies.cause : "<unknown>");
iaxs[fr.callno]->reg->regstate = REG_STATE_REJECTED;
}
+ /* Send ack immediately, before we destroy */
+ send_command_immediate(iaxs[fr.callno], AST_FRAME_IAX, AST_IAX2_COMMAND_ACK, fr.ts, NULL, 0,fr.iseqno);
iax2_destroy_nolock(fr.callno);
break;
case AST_IAX2_COMMAND_REGAUTH: