📄 str_utils.c
字号:
/* The oRTP library is an RTP (Realtime Transport Protocol - rfc1889) stack. Copyright (C) 2001 Simon MORLAT simon.morlat@linphone.org This library is free software; you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation; either version 2.1 of the License, or (at your option) any later version. This library 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 Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with this library; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA*/#include "ortp/ortp.h"#include "ortp/rtp.h"#include "ortp/str_utils.h"#include "utils.h"void qinit(queue_t *q){ mblk_init(&q->_q_stopper); q->_q_stopper.b_next=&q->_q_stopper; q->_q_stopper.b_prev=&q->_q_stopper; q->q_mcount=0;}void mblk_init(mblk_t *mp){ mp->b_cont=mp->b_prev=mp->b_next=NULL; mp->b_rptr=mp->b_wptr=NULL; mp->reserved1=0; mp->reserved2=0;}dblk_t *datab_alloc(int size){ dblk_t *db; int total_size=sizeof(dblk_t)+size; db=(dblk_t *) ortp_malloc(total_size); db->db_base=(uint8_t*)db+sizeof(dblk_t); db->db_lim=db->db_base+size; db->db_ref=1; db->db_freefn=NULL; /* the buffer pointed by db_base must never be freed !*/ return db;}static inline void datab_ref(dblk_t *d){ d->db_ref++;}static inline void datab_unref(dblk_t *d){ d->db_ref--; if (d->db_ref==0){ if (d->db_freefn!=NULL) d->db_freefn(d->db_base); ortp_free(d); }}mblk_t *allocb(int size, int pri){ mblk_t *mp; dblk_t *datab; mp=(mblk_t *) ortp_malloc(sizeof(mblk_t)); mblk_init(mp); datab=datab_alloc(size); mp->b_datap=datab; mp->b_rptr=mp->b_wptr=datab->db_base; mp->b_next=mp->b_prev=mp->b_cont=NULL; return mp;}mblk_t *esballoc(uint8_t *buf, int size, int pri, void (*freefn)(void*) ){ mblk_t *mp; dblk_t *datab; mp=(mblk_t *) ortp_malloc(sizeof(mblk_t)); mblk_init(mp); datab=(dblk_t *) ortp_malloc(sizeof(dblk_t)); datab->db_base=buf; datab->db_lim=buf+size; datab->db_ref=1; datab->db_freefn=freefn; mp->b_datap=datab; mp->b_rptr=mp->b_wptr=buf; mp->b_next=mp->b_prev=mp->b_cont=NULL; return mp;} void freeb(mblk_t *mp){ return_if_fail(mp->b_datap!=NULL); return_if_fail(mp->b_datap->db_base!=NULL); datab_unref(mp->b_datap); ortp_free(mp);}void freemsg(mblk_t *mp){ mblk_t *tmp1,*tmp2; tmp1=mp; while(tmp1!=NULL) { tmp2=tmp1->b_cont; freeb(tmp1); tmp1=tmp2; }}mblk_t *dupb(mblk_t *mp){ mblk_t *newm; return_val_if_fail(mp->b_datap!=NULL,NULL); return_val_if_fail(mp->b_datap->db_base!=NULL,NULL); datab_ref(mp->b_datap); newm=(mblk_t *) ortp_malloc(sizeof(mblk_t)); mblk_init(newm); newm->reserved1=mp->reserved1; newm->reserved2=mp->reserved2; newm->b_datap=mp->b_datap; newm->b_rptr=mp->b_rptr; newm->b_wptr=mp->b_wptr; return newm;}/* duplicates a complex mblk_t */mblk_t *dupmsg(mblk_t* m){ mblk_t *newm=NULL,*mp,*prev; prev=newm=dupb(m); m=m->b_cont; while (m!=NULL){ mp=dupb(m); prev->b_cont=mp; prev=mp; m=m->b_cont; } return newm;}void putq(queue_t *q,mblk_t *mp){ q->_q_stopper.b_prev->b_next=mp; mp->b_prev=q->_q_stopper.b_prev; mp->b_next=&q->_q_stopper; q->_q_stopper.b_prev=mp; q->q_mcount++;}mblk_t *getq(queue_t *q){ mblk_t *tmp; tmp=q->_q_stopper.b_next; if (tmp==&q->_q_stopper) return NULL; q->_q_stopper.b_next=tmp->b_next; tmp->b_next->b_prev=&q->_q_stopper; tmp->b_prev=NULL; tmp->b_next=NULL; q->q_mcount--; return tmp;}mblk_t * peekq(queue_t *q){ mblk_t *tmp; tmp=q->_q_stopper.b_next; if (tmp==&q->_q_stopper) return NULL; return tmp;}/* insert mp in q just before emp */void insq(queue_t *q,mblk_t *emp, mblk_t *mp){ if (emp==NULL){ putq(q,mp); return; } q->q_mcount++; emp->b_prev->b_next=mp; mp->b_prev=emp->b_prev; emp->b_prev=mp; mp->b_next=emp; }void remq(queue_t *q, mblk_t *mp){ q->q_mcount--; mp->b_prev->b_next=mp->b_next; mp->b_next->b_prev=mp->b_prev; mp->b_next=NULL; mp->b_prev=NULL;}/* remove and free all messages in the q */void flushq(queue_t *q, int how){ mblk_t *mp; while ((mp=getq(q))!=NULL) { freemsg(mp); }}int msgdsize(const mblk_t *mp){ int msgsize=0; while(mp!=NULL){ msgsize+=(int) (mp->b_wptr-mp->b_rptr); mp=mp->b_cont; } return msgsize;}void msgpullup(mblk_t *mp,int len){ mblk_t *firstm=mp; dblk_t *db; int wlen=0; if (mp->b_cont==NULL && len==-1) return; /*nothing to do, message is not fragmented */ if (len==-1) len=msgdsize(mp); db=datab_alloc(len); while(wlen<len && mp!=NULL){ int remain=len-wlen; int mlen=mp->b_wptr-mp->b_rptr; if (mlen<=remain){ memcpy(&db->db_base[wlen],mp->b_rptr,mlen); wlen+=mlen; mp=mp->b_cont; }else{ memcpy(&db->db_base[wlen],mp->b_rptr,remain); wlen+=remain; } } /*set firstm to point to the new datab */ freemsg(firstm->b_cont); firstm->b_cont=NULL; datab_unref(firstm->b_datap); firstm->b_datap=db; firstm->b_rptr=db->db_base; firstm->b_wptr=firstm->b_rptr+wlen;}mblk_t *copyb(mblk_t *mp){ mblk_t *newm; int len=(int) (mp->b_wptr-mp->b_rptr); newm=allocb(len,BPRI_MED); memcpy(newm->b_wptr,mp->b_rptr,len); newm->b_wptr+=len; return newm;}mblk_t *copymsg(mblk_t *mp){ mblk_t *newm=0,*m; m=newm=copyb(mp); mp=mp->b_cont; while(mp!=NULL){ m->b_cont=copyb(mp); m=m->b_cont; mp=mp->b_cont; } return newm;}mblk_t * appendb(mblk_t *mp, const char *data, int size, bool_t pad){ int padcnt=0; int i; if (pad){ padcnt= (int)(4L-( (long)(((long)mp->b_wptr)+size) % 4L)) % 4L; } if ((mp->b_wptr + size +padcnt) > mp->b_datap->db_lim){ /* buffer is not large enough: append a new block (with the same size ?)*/ int plen=(int)((char*)mp->b_datap->db_lim - (char*) mp->b_datap->db_base); mp->b_cont=allocb(MAX(plen,size),0); mp=mp->b_cont; } if (size) memcpy(mp->b_wptr,data,size); mp->b_wptr+=size; for (i=0;i<padcnt;i++){ mp->b_wptr[0]=0; mp->b_wptr++; } return mp;}void msgappend(mblk_t *mp, const char *data, int size, bool_t pad){ while(mp->b_cont!=NULL) mp=mp->b_cont; appendb(mp,data,size,pad);}mblk_t *concatb(mblk_t *mp, mblk_t *newm){ while (mp->b_cont!=NULL) mp=mp->b_cont; mp->b_cont=newm; while(newm->b_cont!=NULL) newm=newm->b_cont; return newm;}void msgb_allocator_init(msgb_allocator_t *a){ qinit(&a->q);}mblk_t *msgb_allocator_alloc(msgb_allocator_t *a, int size){ queue_t *q=&a->q; mblk_t *m,*found=NULL; /*lookup for an unused msgb (data block with ref count ==1)*/ for(m=qbegin(q);!qend(q,m);m=qnext(q,m)){ if (m->b_datap->db_ref==1 && m->b_datap->db_lim-m->b_datap->db_base>=size){ found=m; break; } } if (found==NULL){ found=allocb(size,0); putq(q,found); } return dupb(found);}void msgb_allocator_uninit(msgb_allocator_t *a){ flushq(&a->q,-1);}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -