RtspMsg.cxx
上传用户:sy_wanhua
上传日期:2013-07-25
资源大小:3048k
文件大小:10k
源码类别:

流媒体/Mpeg4/MP4

开发平台:

C/C++

  1. /* ====================================================================
  2.  * The Vovida Software License, Version 1.0 
  3.  * 
  4.  * Copyright (c) 2000 Vovida Networks, Inc.  All rights reserved.
  5.  * 
  6.  * Redistribution and use in source and binary forms, with or without
  7.  * modification, are permitted provided that the following conditions
  8.  * are met:
  9.  * 
  10.  * 1. Redistributions of source code must retain the above copyright
  11.  *    notice, this list of conditions and the following disclaimer.
  12.  * 
  13.  * 2. Redistributions in binary form must reproduce the above copyright
  14.  *    notice, this list of conditions and the following disclaimer in
  15.  *    the documentation and/or other materials provided with the
  16.  *    distribution.
  17.  * 
  18.  * 3. The names "VOCAL", "Vovida Open Communication Application Library",
  19.  *    and "Vovida Open Communication Application Library (VOCAL)" must
  20.  *    not be used to endorse or promote products derived from this
  21.  *    software without prior written permission. For written
  22.  *    permission, please contact vocal@vovida.org.
  23.  *
  24.  * 4. Products derived from this software may not be called "VOCAL", nor
  25.  *    may "VOCAL" appear in their name, without prior written
  26.  *    permission of Vovida Networks, Inc.
  27.  * 
  28.  * THIS SOFTWARE IS PROVIDED "AS IS" AND ANY EXPRESSED OR IMPLIED
  29.  * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
  30.  * OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, TITLE AND
  31.  * NON-INFRINGEMENT ARE DISCLAIMED.  IN NO EVENT SHALL VOVIDA
  32.  * NETWORKS, INC. OR ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT DAMAGES
  33.  * IN EXCESS OF 281421,000, NOR FOR ANY INDIRECT, INCIDENTAL, SPECIAL,
  34.  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
  35.  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
  36.  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
  37.  * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  38.  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE
  39.  * USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH
  40.  * DAMAGE.
  41.  * 
  42.  * ====================================================================
  43.  * 
  44.  * This software consists of voluntary contributions made by Vovida
  45.  * Networks, Inc. and many individuals on behalf of Vovida Networks,
  46.  * Inc.  For more information on Vovida Networks, Inc., please see
  47.  * <http://www.vovida.org/>.
  48.  *
  49.  */
  50. static const char* const RtspMsg_cxx_Version = 
  51.     "$Id: RtspMsg.cxx,v 1.18 2001/05/15 20:26:12 bko Exp $";
  52. #include "RtspMsg.hxx"
  53. #include "CharDataParser.hxx"
  54. #include "RtspBadDataException.hxx"
  55. #include "cpLog.h"
  56. RtspMsg::RtspMsg()
  57.     : myTransConnPtr(0),
  58.       myStartLine(),
  59.       myHeaders(),
  60.       myNumHeader(0),
  61.       myContentLength(0),
  62.       myHasBody(false),
  63.       myMsgBody(),
  64.       myCompletionFlag(RTSP_MSG_NEW),
  65.       myIncompHeader(),
  66.       myFirstWordUnknown(false),
  67.       mySessionId(),
  68.       myCSeq(),
  69.       myContentBase(),
  70.       myContentType(),
  71.       myTransportSpec(0),
  72.       myRange(0)
  73. {
  74.     for(u_int32_t i = 0; i < RTSP_UNKNOWN_HDR; i++)
  75.     {
  76.         myHeaderMap[i].offset = -1;
  77.         myHeaderMap[i].len = 0;
  78.     }
  79. }
  80. RtspMsg::RtspMsg(const RtspMsg& src)
  81.     : myTransConnPtr(src.myTransConnPtr),
  82.       myStartLine(src.myStartLine),
  83.       myHeaders(src.myHeaders),
  84.       myNumHeader(src.myNumHeader),
  85.       myContentLength(src.myContentLength),
  86.       myHasBody(src.myHasBody),
  87.       myMsgBody(src.myMsgBody),
  88.       myCompletionFlag(src.myCompletionFlag),
  89.       myIncompHeader(src.myIncompHeader),
  90.       myFirstWordUnknown(src.myFirstWordUnknown),
  91.       mySessionId(src.mySessionId),
  92.       myCSeq(src.myCSeq),
  93.       myContentBase(src.myContentBase),
  94.       myContentType(src.myContentType),
  95.       myTransportSpec(0),
  96.       myRange(0)
  97. {
  98.     if (src.myTransportSpec != 0)
  99.     {
  100.         myTransportSpec = new RtspTransportSpec(*src.myTransportSpec);
  101.     }
  102.     if (src.myRange != 0)
  103.     {
  104.         myRange = new RtspRangeHdr(*src.myRange);
  105.     }
  106.     for(u_int32_t i = 0; i < RTSP_UNKNOWN_HDR; i++)
  107.     {
  108.         myHeaderMap[i].offset = src.myHeaderMap[i].offset;
  109.         myHeaderMap[i].len = src.myHeaderMap[i].len;
  110.     }
  111. }
  112. RtspMsg&
  113. RtspMsg::operator=(const RtspMsg& src)
  114. {
  115.     if (&src != this)
  116.     {
  117.         myTransConnPtr = src.myTransConnPtr;
  118.         myStartLine = src.myStartLine;
  119.         myHeaders = src.myHeaders;
  120.         myNumHeader = src.myNumHeader;
  121.         myContentLength = src.myContentLength;
  122.         myHasBody = src.myHasBody;
  123.         myMsgBody = src.myMsgBody;
  124.         myCompletionFlag = src.myCompletionFlag;
  125.         myIncompHeader = src.myIncompHeader;
  126.         myFirstWordUnknown = src.myFirstWordUnknown;
  127.         mySessionId = src.mySessionId;
  128.         myCSeq = src.myCSeq;
  129.         myContentBase = src.myContentBase;
  130.         myContentType = src.myContentType;
  131.         myTransportSpec = 0;
  132.         myRange = 0;
  133.         if (src.myTransportSpec != 0)
  134.         {
  135.             myTransportSpec = new RtspTransportSpec(*src.myTransportSpec);
  136.         }
  137.         if (src.myRange != 0)
  138.         {
  139.             myRange = new RtspRangeHdr(*src.myRange);
  140.         }
  141.         for(u_int32_t i = 0; i < RTSP_UNKNOWN_HDR; i++)
  142.         {
  143.             myHeaderMap[i].offset = src.myHeaderMap[i].offset;
  144.             myHeaderMap[i].len = src.myHeaderMap[i].len;
  145.         }
  146.     }
  147.     return *this;
  148. }
  149. const u_int32_t
  150. RtspMsg::getHdrBodyValue(RtspHeadersType header) const
  151. {
  152.     u_int32_t theValue = 0;
  153.     if (myHeaderMap[header].offset != -1)
  154.     {
  155.         CharData numData;
  156. //        cpLog(LOG_DEBUG_STACK, "myHeaders is: %s", myHeaders.getData());
  157.         numData.set(myHeaders.getDataBuf() + 
  158.                     myHeaderMap[header].offset,
  159.                     myHeaderMap[header].len);
  160. //        cpLog(LOG_DEBUG_STACK, "The hdr body offset is: %d", 
  161. //              myHeaderMap[header].offset);
  162. //        cpLog(LOG_DEBUG_STACK, "The hdr body len is: %d", numData.getLen());
  163.         CharDataParser numDataParser(&numData);
  164.         u_int32_t value;
  165.         if ( numDataParser.getNextInteger(value) )
  166.         {
  167.             theValue = value;
  168.         }
  169.         else
  170.         {
  171.             cpLog(LOG_ERR, "error read value of header: %d", header);
  172.         }
  173.     }
  174.     return theValue;
  175. }
  176. const Data
  177. RtspMsg::getHdrBodyData(RtspHeadersType header, bool full) const
  178. {
  179.     Data theData;
  180.     if (myHeaderMap[header].offset != -1)
  181.     {
  182.         CharData bodyData;
  183.         bodyData.set(myHeaders.getDataBuf() + 
  184.                      myHeaderMap[header].offset,
  185.                      myHeaderMap[header].len);
  186.         CharDataParser bodyParser(&bodyData);
  187.         CharData theFirstBody;
  188.         u_int8_t* stop = (full) ? bodyParser.myMaskEol : 
  189.                                   bodyParser.myMaskEolSpace;
  190.         if (bodyParser.parseUntil(&theFirstBody, stop))
  191.         {
  192.             Data theFirstData(theFirstBody.getPtr(), theFirstBody.getLen());
  193.             theData = theFirstData;
  194.         }
  195.         else
  196.         {
  197.             cpLog(LOG_ERR, "error read data of header: %d", header);
  198.         }
  199.     }
  200.     return theData;
  201. }
  202. const int
  203. RtspMsg::getContentLength() 
  204. {
  205.     if (myContentLength == 0)
  206.     {
  207.         myContentLength = getHdrBodyValue(RTSP_CONTENT_LENGTH_HDR);
  208.     }
  209.     return myContentLength;
  210. }
  211. bool
  212. RtspMsg::isValid()
  213. {
  214.     if (myHeaderMap[RTSP_CSEQ_HDR].offset != -1)
  215.         return true;
  216.     else
  217.         return false;
  218. }
  219. bool
  220. RtspMsg::isRequest()
  221. {
  222.     CharData strtLine;
  223.     strtLine.set(myStartLine.getDataBuf(), myStartLine.length());
  224.     CharDataParser charParser(&strtLine);
  225.     CharData firstWord;
  226.     charParser.getNextWord(&firstWord);
  227.     return ( !(firstWord.isEqualNoCase("RTSP", 4)) );
  228. }
  229. bool
  230. RtspMsg::isResponse()
  231. {
  232.     return ( !(isRequest()) );
  233. }
  234. const Data& 
  235. RtspMsg::getSessionId()
  236. {
  237.     if (mySessionId.length() == 0)
  238.     {
  239.         mySessionId = getHdrBodyData(RTSP_SESSION_HDR, false);
  240.         cpLog(LOG_DEBUG_STACK, "SessionId is: %s", mySessionId.getData() );
  241.     }
  242.     return mySessionId;
  243. }
  244. const Data&
  245. RtspMsg::getCSeq()
  246. {
  247.     if (myCSeq.length() == 0)
  248.     {
  249.         myCSeq = getHdrBodyData(RTSP_CSEQ_HDR, false);
  250.     }
  251.     return myCSeq;
  252. }
  253. const Data&
  254. RtspMsg::getContentBase()
  255. {
  256.     if (myContentBase.length() == 0)
  257.     {
  258.         myContentBase = getHdrBodyData(RTSP_CONTENT_BASE_HDR, false);
  259.     }
  260.     return myContentBase;
  261. }
  262. const Data&
  263. RtspMsg::getContentType()
  264. {
  265.     if (myContentType.length() == 0)
  266.     {
  267.         myContentType = getHdrBodyData(RTSP_CONTENT_TYPE_HDR, true);
  268.     }
  269.     return myContentType;
  270. }
  271. Sptr< RtspTransportSpec >
  272. RtspMsg::getTransport()
  273. {
  274.     if (myTransportSpec == 0)
  275.     {
  276.         if (myHeaderMap[RTSP_TRANSPORT_HDR].offset != -1)
  277.         {
  278.             CharData transpData;
  279.             transpData.set(myHeaders.getDataBuf() + 
  280.                            myHeaderMap[RTSP_TRANSPORT_HDR].offset,
  281.                            myHeaderMap[RTSP_TRANSPORT_HDR].len);
  282.             CharDataParser transpParser(&transpData);
  283.     
  284.             RtspTransportHdr transHdr;
  285.             myTransportSpec = transHdr.parseTransportHeader(transpParser);
  286.         }
  287.     }
  288.     return myTransportSpec;
  289. }
  290. Sptr< RtspRangeHdr >
  291. RtspMsg::getRange()
  292. {
  293.     if (myRange == 0)
  294.     {
  295.         if (myHeaderMap[RTSP_RANGE_HDR].offset != -1)
  296.         {
  297.             CharData rangeData;
  298.             rangeData.set(myHeaders.getDataBuf() + 
  299.                           myHeaderMap[RTSP_RANGE_HDR].offset,
  300.                           myHeaderMap[RTSP_RANGE_HDR].len);
  301.             CharDataParser rangeParser(&rangeData);
  302.             myRange = new RtspRangeHdr();
  303.             myRange->parseRangeHeader(rangeParser);
  304.         }
  305.     }
  306.     return myRange;
  307. }
  308. /* Local Variables: */
  309. /* c-file-style: "stroustrup" */
  310. /* indent-tabs-mode: nil */
  311. /* c-file-offsets: ((access-label . -) (inclass . ++)) */
  312. /* c-basic-offset: 4 */
  313. /* End: */