⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 sipfrag.cxx

📁 一个著名的SIP协议栈
💻 CXX
字号:
#if defined(HAVE_CONFIG_H)
#include "resip/stack/config.hxx"
#endif

#include "resip/stack/MsgHeaderScanner.hxx"
#include "resip/stack/SipFrag.hxx"
#include "resip/stack/SipMessage.hxx"
#include "rutil/Logger.hxx"
#include "rutil/ParseBuffer.hxx"
#include "rutil/WinLeakCheck.hxx"

using namespace resip;
using namespace std;

#define RESIPROCATE_SUBSYSTEM Subsystem::SIP

bool
SipFrag::init()
{
   static ContentsFactory<SipFrag> factory;
   (void)factory;
   return true;
}

SipFrag::SipFrag(const Mime& contentsType)
   : Contents(contentsType),
     mMessage(new SipMessage())
{}

SipFrag::SipFrag(HeaderFieldValue* hfv, const Mime& contentsType)
   : Contents(hfv, contentsType),
     mMessage(0)
{
}

SipFrag::SipFrag(const SipFrag& rhs)
   : Contents(rhs,HeaderFieldValue::CopyPadding),
     mMessage(rhs.mMessage ? new SipMessage(*rhs.mMessage) : 0)
{
}

SipFrag::~SipFrag()
{
   delete mMessage;
}

SipFrag&
SipFrag::operator=(const SipFrag& rhs)
{
   if (this != &rhs)
   {
      Contents::operator=(rhs);
      delete mMessage;
      if (rhs.mMessage)
      {
         mMessage = new SipMessage(*rhs.mMessage);
      }
      else
      {
         mMessage = 0;
      }
   }
   
   return *this;
}

Contents* 
SipFrag::clone() const
{
   return new SipFrag(*this);
}

const Mime& 
SipFrag::getStaticType() 
{
   static Mime type("message", "sipfrag");
   //static Mime type("application", "sipfrag");
   return type;
}

SipMessage& 
SipFrag::message() 
{
   checkParsed(); 
   return *mMessage;
}

const SipMessage& 
SipFrag::message() const 
{
   checkParsed(); 
   return *mMessage;
}

std::ostream& 
SipFrag::encodeParsed(std::ostream& str) const
{
   mMessage->encodeSipFrag(str);

   return str;
}

bool 
SipFrag::hasStartLine(char* buffer, int size)
{
#if 0
   //!dcm! -- this probably inefficient, but the SIP grammer makes this very
   //difficult. Better here than in the MsgHeaderScanner. There's also proabably a
   //way to make a header that matches the requestLine check which isn't a
   //request line.

   ParseBuffer pbCheck(buffer, size);
   pbCheck.skipWhitespace(); //gratuitous?

   //!dcm! -- could extend to SIP/2.0, but nobody should start a hname with SIP/
   if ((pbCheck.end() - pbCheck.position()) > 4 &&
      strncmp(pbCheck.position(), "SIP/", 4) == 0)
   {
      return true;
   }
   else
   {
      pbCheck.skipToChars(Symbols::CRLF);
      if (pbCheck.eof()) 
      {
         //false positive, let MsgHeaderScanner sort the exact error out
         return true; 
      }
      
      pbCheck.skipBackToChar(Symbols::SPACE[0]);
      if (pbCheck.position() == pbCheck.start()) 
      {
         return false;
      }
         
      if ((pbCheck.end() - pbCheck.position()) > 4 &&
          strncmp(pbCheck.position(), "SIP/", 4) == 0)
      {
         return true;
      }
      else
      {
         return false;
      }
   }
#else   
   //!dcm! -- better approach, remove above if this is proven to be correct
   ParseBuffer pbCheck(buffer, size);
   pbCheck.skipWhitespace(); //gratuitous?
   pbCheck.skipToOneOf(" \t:\r\n");
   while(!pbCheck.eof())
   {
      switch(*pbCheck.position())
      {
         case ':':
            return false;
         case ' ':
         case '\t':
            pbCheck.skipChar();
            break;             
         case '\r':
         case '\n':
            return false;
         default:
            return true;
      }
   }
   return true;  //false positive, let MsgHeaderScanner sort the exact error out
#endif
}

void 
SipFrag::parse(ParseBuffer& pb)
{
//   DebugLog(<< "SipFrag::parse: " << pb.position());

   mMessage = new SipMessage();

   pb.assertNotEof();
   const char *constBuffer = pb.position();
   char *buffer = const_cast<char *>(constBuffer);

   size_t size = pb.end() - pb.position();

   // !ah! removed size check .. process() cannot process more
   // than size bytes of the message.


   MsgHeaderScanner msgHeaderScanner;
   msgHeaderScanner.prepareForFrag(mMessage, hasStartLine(buffer, size));
   enum { sentinelLength = 4 };  // Two carriage return / line feed pairs.
   //char saveTermCharArray[sentinelLength];
   static char* sentinel="\r\n\r\n";
   char *termCharArray = buffer + size;
   memcpy(scratchpad,termCharArray,4);
   
   /*saveTermCharArray[0] = termCharArray[0];
   saveTermCharArray[1] = termCharArray[1];
   saveTermCharArray[2] = termCharArray[2];
   saveTermCharArray[3] = termCharArray[3];*/
   
   memcpy(termCharArray,sentinel,4);
   /*termCharArray[0] = '\r';
   termCharArray[1] = '\n';
   termCharArray[2] = '\r';
   termCharArray[3] = '\n';*/
   char *scanTermCharPtr;
   MsgHeaderScanner::ScanChunkResult scanChunkResult =
       msgHeaderScanner.scanChunk(buffer,
                                  size + sentinelLength,
                                  &scanTermCharPtr);
   
   memcpy(termCharArray,scratchpad,4);
   /*termCharArray[0] = saveTermCharArray[0];
   termCharArray[1] = saveTermCharArray[1];
   termCharArray[2] = saveTermCharArray[2];
   termCharArray[3] = saveTermCharArray[3];*/
   
   // !dlb! not at all clear what to do here
   // see: "// tests end of message problem (MsgHeaderScanner?)"
   //      in test/testSipFrag.cxx
   if (false && scanChunkResult != MsgHeaderScanner::scrEnd) 
   {
      CerrLog(<< "not MsgHeaderScanner::scrEnd");
      pb.fail(__FILE__, __LINE__);
   } 
   else 
   {
      size_t used = scanTermCharPtr - buffer;

      // !ah! I think this is broken .. if we are UDP then the 
      // remainder is the SigFrag, not the Content-Length... ??
      if (mMessage->exists(h_ContentLength))
      {
         mMessage->setBody(scanTermCharPtr,
                           static_cast<int>(size - used));
      }
      else
      {
         // !ah! So the headers weren't complete. Why are we here?
         // !dlb! 
         if (mMessage->exists(h_ContentLength))
         {
            pb.reset(buffer + used);
            pb.skipChars(Symbols::CRLF);
            mMessage->setBody(pb.position(),int(pb.end()-pb.position()) );
         }
      }
      pb.reset(pb.end());
   }
}

/* ====================================================================
 * The Vovida Software License, Version 1.0 
 * 
 * Copyright (c) 2000 Vovida Networks, Inc.  All rights reserved.
 * 
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 * 
 * 1. Redistributions of source code must retain the above copyright
 *    notice, this list of conditions and the following disclaimer.
 * 
 * 2. Redistributions in binary form must reproduce the above copyright
 *    notice, this list of conditions and the following disclaimer in
 *    the documentation and/or other materials provided with the
 *    distribution.
 * 
 * 3. The names "VOCAL", "Vovida Open Communication Application Library",
 *    and "Vovida Open Communication Application Library (VOCAL)" must
 *    not be used to endorse or promote products derived from this
 *    software without prior written permission. For written
 *    permission, please contact vocal@vovida.org.
 *
 * 4. Products derived from this software may not be called "VOCAL", nor
 *    may "VOCAL" appear in their name, without prior written
 *    permission of Vovida Networks, Inc.
 * 
 * THIS SOFTWARE IS PROVIDED "AS IS" AND ANY EXPRESSED OR IMPLIED
 * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
 * OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, TITLE AND
 * NON-INFRINGEMENT ARE DISCLAIMED.  IN NO EVENT SHALL VOVIDA
 * NETWORKS, INC. OR ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT DAMAGES
 * IN EXCESS OF $1,000, NOR FOR ANY INDIRECT, INCIDENTAL, SPECIAL,
 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
 * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE
 * USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH
 * DAMAGE.
 * 
 * ====================================================================
 * 
 * This software consists of voluntary contributions made by Vovida
 * Networks, Inc. and many individuals on behalf of Vovida Networks,
 * Inc.  For more information on Vovida Networks, Inc., please see
 * <http://www.vovida.org/>.
 *
 */

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -