shmhelp.cpp
上传用户:dangjiwu
上传日期:2013-07-19
资源大小:42019k
文件大小:7k
源码类别:

Symbian

开发平台:

Visual C++

  1. /* ***** BEGIN LICENSE BLOCK *****
  2.  * Source last modified: $Id: shmhelp.cpp,v 1.2.4.2 2004/07/09 12:48:45 pankajgupta Exp $
  3.  * 
  4.  * Portions Copyright (c) 1995-2004 RealNetworks, Inc. All Rights Reserved.
  5.  * 
  6.  * The contents of this file, and the files included with this file,
  7.  * are subject to the current version of the RealNetworks Public
  8.  * Source License (the "RPSL") available at
  9.  * http://www.helixcommunity.org/content/rpsl unless you have licensed
  10.  * the file under the current version of the RealNetworks Community
  11.  * Source License (the "RCSL") available at
  12.  * http://www.helixcommunity.org/content/rcsl, in which case the RCSL
  13.  * will apply. You may also obtain the license terms directly from
  14.  * RealNetworks.  You may not use this file except in compliance with
  15.  * the RPSL or, if you have a valid RCSL with RealNetworks applicable
  16.  * to this file, the RCSL.  Please see the applicable RPSL or RCSL for
  17.  * the rights, obligations and limitations governing use of the
  18.  * contents of the file.
  19.  * 
  20.  * Alternatively, the contents of this file may be used under the
  21.  * terms of the GNU General Public License Version 2 or later (the
  22.  * "GPL") in which case the provisions of the GPL are applicable
  23.  * instead of those above. If you wish to allow use of your version of
  24.  * this file only under the terms of the GPL, and not to allow others
  25.  * to use your version of this file under the terms of either the RPSL
  26.  * or RCSL, indicate your decision by deleting the provisions above
  27.  * and replace them with the notice and other provisions required by
  28.  * the GPL. If you do not delete the provisions above, a recipient may
  29.  * use your version of this file under the terms of any one of the
  30.  * RPSL, the RCSL or the GPL.
  31.  * 
  32.  * This file is part of the Helix DNA Technology. RealNetworks is the
  33.  * developer of the Original Code and owns the copyrights in the
  34.  * portions it created.
  35.  * 
  36.  * This file, and the files included with this file, is distributed
  37.  * and made available on an 'AS IS' basis, WITHOUT WARRANTY OF ANY
  38.  * KIND, EITHER EXPRESS OR IMPLIED, AND REALNETWORKS HEREBY DISCLAIMS
  39.  * ALL SUCH WARRANTIES, INCLUDING WITHOUT LIMITATION, ANY WARRANTIES
  40.  * OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, QUIET
  41.  * ENJOYMENT OR NON-INFRINGEMENT.
  42.  * 
  43.  * Technology Compatibility Kit Test Suite(s) Location:
  44.  *    http://www.helixcommunity.org/content/tck
  45.  * 
  46.  * Contributor(s):
  47.  * 
  48.  * ***** END LICENSE BLOCK ***** */
  49. // for shared memory
  50. #include <sys/types.h>
  51. #include <sys/ipc.h>
  52. #include <sys/shm.h>
  53. #include <sys/utsname.h>
  54. #include "hlxclib/string.h"
  55. #include "shmhelp.h"
  56. #include "hxassert.h"
  57. //Work around AIX problem.
  58. #ifdef _AIX
  59. #  define MAX_SHARED_REGIONS ((UINT32)1)
  60. #else
  61. #  define MAX_SHARED_REGIONS ((UINT32)9999)
  62. #endif
  63. //Keeps track of how many shared memory segments we have used.
  64. int      ShmHelp::zm_nSegment = 0;
  65. Display* ShmHelp::zm_pDisplay = NULL;
  66. BOOL     ShmHelp::zm_bUseShm  = FALSE;
  67. void ShmHelp::Init(Display* pDisplay)
  68. {
  69.    HX_ASSERT( pDisplay );
  70.    zm_pDisplay = pDisplay;
  71.    zm_bUseShm  = FALSE;
  72.    
  73.    //See if our XServer supports ShmMemory....
  74. #if defined(_AIX)  
  75.    int majorVersion, minorVersion;
  76.    Bool sharedPixmaps;
  77.    // doesn't seem to work with multiple presentation panes...
  78.    XLockDisplay(zm_pDisplay);
  79.    zm_bUseShm = XShmQueryVersion( zm_pDisplay,
  80.                                   &majorVersion,
  81.                                   &minorVersion, &sharedPixmaps); 
  82.    XUnlockDisplay(zm_pDisplay);
  83.    if (zm_nSegment > AIX_MAX_SHARED_REGIONS) 
  84.       zm_bUseShm = FALSE;
  85. #else
  86. #  ifdef _SOLARIS
  87.    zm_bUseShm = FALSE;
  88. #  else
  89.    XLockDisplay(zm_pDisplay);   
  90.    zm_bUseShm = XShmQueryExtension(zm_pDisplay);
  91.    XUnlockDisplay(zm_pDisplay);
  92. #  endif    
  93. #endif    
  94.    //Make sure we only use it if we are displaying locally.  The
  95.    //DISPLAY string can start with ':', '0' or the machine name.
  96.    if(zm_bUseShm)
  97.    {
  98.       char *disp = getenv("DISPLAY");
  99.       HX_ASSERT( disp != NULL && "Your display variable isn't set!" );
  100.       if(!disp)
  101.       {
  102.          zm_bUseShm = FALSE;
  103.       }
  104.       else if( disp[0]!=':' && disp[0]!='0' )
  105.       {
  106.          //See if our machine name is in the dipsplay string.
  107.          struct utsname inf;
  108.          uname(&inf);
  109.          if( (strlen(disp)<=strlen(inf.machine)) ||
  110.              (strncmp(disp, inf.machine, strlen(inf.machine))) )
  111.          {
  112.             zm_bUseShm = FALSE;
  113.          }
  114.       }
  115.    }
  116. }
  117. //This ctor implies someone has called the one that sets the
  118. //display var.
  119. ShmHelp::ShmHelp()
  120. {
  121.    HX_ASSERT( zm_pDisplay );
  122. }
  123. BOOL ShmHelp::ShmAvailable()
  124. {
  125.    return zm_bUseShm;
  126. }
  127. HX_RESULT ShmHelp::CreateSharedRegion(INT32 nSize, UCHAR**ppMem, int* pnShmID, XShmSegmentInfo* pInfo ) 
  128. {
  129.     HX_RESULT theErr   = HXR_OK;
  130.     int       nRetCode = 0;
  131.     HX_ASSERT( ppMem );
  132.     HX_ASSERT( pnShmID );
  133.     //Make sure we were inited...
  134.     HX_ASSERT(zm_pDisplay);
  135.     if( !zm_pDisplay )
  136.        return HXR_UNEXPECTED;
  137.     *ppMem   = NULL;
  138.     *pnShmID = -1;
  139.     
  140.     //get the shared memory segment, etc.
  141.     if( !zm_bUseShm ) 
  142.     {
  143. theErr = HXR_UNEXPECTED;
  144.     }
  145.     
  146. #ifdef _AIX
  147.     if (zm_nSegment > AIX_MAX_SHARED_REGIONS)
  148.     {
  149. theErr = HXR_UNEXPECTED;
  150.     }
  151. #endif
  152.     if( theErr == HXR_OK )
  153.     {
  154.         // Request a new shared memory ID
  155.         *pnShmID = ::shmget(0, nSize, IPC_CREAT | 0777);
  156.     
  157.         if( *pnShmID == -1 ) 
  158.         {
  159. #ifdef _DEBUG
  160.             perror("shmget");
  161. #endif            
  162.             theErr = HXR_FAIL;
  163.         }
  164.         else
  165.         {
  166.             // Attach shared memory segment to process's address space
  167.             *ppMem = (UCHAR*)::shmat(*pnShmID, 0, 0);
  168.             if( ((int)*ppMem) == -1) 
  169.             {
  170.                 theErr = HXR_FAIL;
  171. #ifdef _DEBUG            
  172.                 perror("shmat");
  173. #endif            
  174.             }
  175.             else
  176.             {
  177.                 //Request that segment be removed after last reference to it 
  178.                 //is released
  179.                 pInfo->shmseg   = ++zm_nSegment;
  180.                 pInfo->shmid    = *pnShmID;
  181.                 pInfo->shmaddr  = (char*)*ppMem;
  182.                 pInfo->readOnly = True;
  183. XLockDisplay(zm_pDisplay);
  184.                 nRetCode = XShmAttach(zm_pDisplay, pInfo);
  185.                 XSync(zm_pDisplay, False);
  186. XUnlockDisplay(zm_pDisplay);
  187.                 ::shmctl(*pnShmID, IPC_RMID, NULL); 
  188.     
  189.                 if( !nRetCode )
  190.                 {
  191.                     theErr = HXR_FAIL;
  192.                 }
  193.                 else
  194.                 {
  195.                     theErr = HXR_OK;
  196.                 }
  197.             }
  198.         }
  199.         if( HXR_OK != theErr )
  200.         {
  201.             // in case of failure, make a best case effort to detach the region
  202.             ::shmctl(*pnShmID, IPC_RMID, NULL);
  203.             zm_bUseShm = FALSE;
  204.         }
  205.     }
  206.     return theErr;
  207. }
  208. HX_RESULT ShmHelp::DetachSharedRegion(UCHAR**ppMem, XShmSegmentInfo* pInfo ) 
  209. {
  210.    HX_ASSERT(zm_bUseShm);
  211.    HX_ASSERT(pInfo);
  212.    HX_ASSERT(*ppMem);
  213.    
  214.    //Make sure we were inited...
  215.    HX_ASSERT(zm_pDisplay);
  216.    if( !zm_pDisplay )
  217.       return HXR_UNEXPECTED;
  218.    
  219.    if(!zm_bUseShm)
  220.       return HXR_UNEXPECTED;
  221.    XLockDisplay(zm_pDisplay);
  222.    XShmDetach(zm_pDisplay, pInfo);
  223.    XUnlockDisplay(zm_pDisplay);
  224.    
  225.    ::shmdt((char*)*ppMem);
  226.    *ppMem = NULL;
  227.    
  228.    --zm_nSegment;
  229.    
  230.    return HXR_OK;
  231. }