xref: /aoo4110/main/vcl/unx/headless/svpelement.cxx (revision b1cdbd2c)
1*b1cdbd2cSJim Jagielski /**************************************************************
2*b1cdbd2cSJim Jagielski  *
3*b1cdbd2cSJim Jagielski  * Licensed to the Apache Software Foundation (ASF) under one
4*b1cdbd2cSJim Jagielski  * or more contributor license agreements.  See the NOTICE file
5*b1cdbd2cSJim Jagielski  * distributed with this work for additional information
6*b1cdbd2cSJim Jagielski  * regarding copyright ownership.  The ASF licenses this file
7*b1cdbd2cSJim Jagielski  * to you under the Apache License, Version 2.0 (the
8*b1cdbd2cSJim Jagielski  * "License"); you may not use this file except in compliance
9*b1cdbd2cSJim Jagielski  * with the License.  You may obtain a copy of the License at
10*b1cdbd2cSJim Jagielski  *
11*b1cdbd2cSJim Jagielski  *   http://www.apache.org/licenses/LICENSE-2.0
12*b1cdbd2cSJim Jagielski  *
13*b1cdbd2cSJim Jagielski  * Unless required by applicable law or agreed to in writing,
14*b1cdbd2cSJim Jagielski  * software distributed under the License is distributed on an
15*b1cdbd2cSJim Jagielski  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16*b1cdbd2cSJim Jagielski  * KIND, either express or implied.  See the License for the
17*b1cdbd2cSJim Jagielski  * specific language governing permissions and limitations
18*b1cdbd2cSJim Jagielski  * under the License.
19*b1cdbd2cSJim Jagielski  *
20*b1cdbd2cSJim Jagielski  *************************************************************/
21*b1cdbd2cSJim Jagielski 
22*b1cdbd2cSJim Jagielski 
23*b1cdbd2cSJim Jagielski 
24*b1cdbd2cSJim Jagielski #include "svpelement.hxx"
25*b1cdbd2cSJim Jagielski 
26*b1cdbd2cSJim Jagielski #include <basebmp/scanlineformats.hxx>
27*b1cdbd2cSJim Jagielski #include <tools/debug.hxx>
28*b1cdbd2cSJim Jagielski 
29*b1cdbd2cSJim Jagielski #if defined WITH_SVP_LISTENING
30*b1cdbd2cSJim Jagielski #include <osl/thread.h>
31*b1cdbd2cSJim Jagielski #include <vcl/svapp.hxx>
32*b1cdbd2cSJim Jagielski #include <rtl/strbuf.hxx>
33*b1cdbd2cSJim Jagielski #include <vcl/bitmap.hxx>
34*b1cdbd2cSJim Jagielski #include <tools/stream.hxx>
35*b1cdbd2cSJim Jagielski 
36*b1cdbd2cSJim Jagielski #include "svpvd.hxx"
37*b1cdbd2cSJim Jagielski #include "svpbmp.hxx"
38*b1cdbd2cSJim Jagielski #include "svpframe.hxx"
39*b1cdbd2cSJim Jagielski 
40*b1cdbd2cSJim Jagielski #include <list>
41*b1cdbd2cSJim Jagielski #include <hash_map>
42*b1cdbd2cSJim Jagielski 
43*b1cdbd2cSJim Jagielski #include <sys/types.h>
44*b1cdbd2cSJim Jagielski #include <sys/socket.h>
45*b1cdbd2cSJim Jagielski #include <netinet/ip.h>
46*b1cdbd2cSJim Jagielski #include <stdio.h>
47*b1cdbd2cSJim Jagielski #include <errno.h>
48*b1cdbd2cSJim Jagielski 
49*b1cdbd2cSJim Jagielski using namespace basegfx;
50*b1cdbd2cSJim Jagielski 
51*b1cdbd2cSJim Jagielski class SvpElementContainer
52*b1cdbd2cSJim Jagielski {
53*b1cdbd2cSJim Jagielski     std::list< SvpElement* >        m_aElements;
54*b1cdbd2cSJim Jagielski     int                             m_nSocket;
55*b1cdbd2cSJim Jagielski     oslThread                       m_aThread;
56*b1cdbd2cSJim Jagielski 
57*b1cdbd2cSJim Jagielski     SvpElementContainer();
58*b1cdbd2cSJim Jagielski     ~SvpElementContainer();
59*b1cdbd2cSJim Jagielski     public:
registerElement(SvpElement * pElement)60*b1cdbd2cSJim Jagielski     void registerElement( SvpElement* pElement ) { m_aElements.push_back(pElement); }
deregisterElement(SvpElement * pElement)61*b1cdbd2cSJim Jagielski     void deregisterElement( SvpElement* pElement ) { m_aElements.remove(pElement); }
62*b1cdbd2cSJim Jagielski 
63*b1cdbd2cSJim Jagielski     void run();
64*b1cdbd2cSJim Jagielski     DECL_LINK(processRequest,void*);
65*b1cdbd2cSJim Jagielski 
66*b1cdbd2cSJim Jagielski     static SvpElementContainer& get();
67*b1cdbd2cSJim Jagielski };
68*b1cdbd2cSJim Jagielski 
69*b1cdbd2cSJim Jagielski extern "C" void SAL_CALL SvpContainerThread( void* );
70*b1cdbd2cSJim Jagielski 
get()71*b1cdbd2cSJim Jagielski SvpElementContainer& SvpElementContainer::get()
72*b1cdbd2cSJim Jagielski {
73*b1cdbd2cSJim Jagielski     static SvpElementContainer* pInstance = new SvpElementContainer();
74*b1cdbd2cSJim Jagielski     return *pInstance;
75*b1cdbd2cSJim Jagielski }
76*b1cdbd2cSJim Jagielski 
SvpElementContainer()77*b1cdbd2cSJim Jagielski SvpElementContainer::SvpElementContainer()
78*b1cdbd2cSJim Jagielski {
79*b1cdbd2cSJim Jagielski     static const char* pEnv = getenv("SVP_LISTENER_PORT");
80*b1cdbd2cSJim Jagielski     int nPort = (pEnv && *pEnv) ? atoi(pEnv) : 8000;
81*b1cdbd2cSJim Jagielski     m_nSocket = socket( PF_INET, SOCK_STREAM, 0 );
82*b1cdbd2cSJim Jagielski     if( m_nSocket >= 0)
83*b1cdbd2cSJim Jagielski     {
84*b1cdbd2cSJim Jagielski         int nOn = 1;
85*b1cdbd2cSJim Jagielski         if( setsockopt(m_nSocket, SOL_SOCKET, SO_REUSEADDR,
86*b1cdbd2cSJim Jagielski                        (char*)&nOn, sizeof(nOn)) )
87*b1cdbd2cSJim Jagielski         {
88*b1cdbd2cSJim Jagielski             perror( "SvpElementContainer: changing socket options failed" );
89*b1cdbd2cSJim Jagielski             close( m_nSocket );
90*b1cdbd2cSJim Jagielski         }
91*b1cdbd2cSJim Jagielski         else
92*b1cdbd2cSJim Jagielski         {
93*b1cdbd2cSJim Jagielski             struct sockaddr_in addr;
94*b1cdbd2cSJim Jagielski             memset(&addr, 0, sizeof(struct sockaddr_in));
95*b1cdbd2cSJim Jagielski             addr.sin_family = AF_INET;
96*b1cdbd2cSJim Jagielski             addr.sin_port = htons(nPort);
97*b1cdbd2cSJim Jagielski             addr.sin_addr.s_addr = INADDR_ANY;
98*b1cdbd2cSJim Jagielski             if( bind(m_nSocket,(struct sockaddr*)&addr,sizeof(addr)) )
99*b1cdbd2cSJim Jagielski             {
100*b1cdbd2cSJim Jagielski                 perror( "SvpElementContainer: bind() failed" );
101*b1cdbd2cSJim Jagielski                 close( m_nSocket );
102*b1cdbd2cSJim Jagielski             }
103*b1cdbd2cSJim Jagielski             else
104*b1cdbd2cSJim Jagielski             {
105*b1cdbd2cSJim Jagielski                 if( listen( m_nSocket, 0 ) )
106*b1cdbd2cSJim Jagielski                 {
107*b1cdbd2cSJim Jagielski                     perror( "SvpElementContainer: listen() failed" );
108*b1cdbd2cSJim Jagielski                     close(m_nSocket);
109*b1cdbd2cSJim Jagielski                 }
110*b1cdbd2cSJim Jagielski                 else
111*b1cdbd2cSJim Jagielski                 {
112*b1cdbd2cSJim Jagielski                     m_aThread = osl_createThread( SvpContainerThread, this );
113*b1cdbd2cSJim Jagielski                 }
114*b1cdbd2cSJim Jagielski             }
115*b1cdbd2cSJim Jagielski         }
116*b1cdbd2cSJim Jagielski     }
117*b1cdbd2cSJim Jagielski     else
118*b1cdbd2cSJim Jagielski         perror( "SvpElementContainer: socket() failed\n" );
119*b1cdbd2cSJim Jagielski }
120*b1cdbd2cSJim Jagielski 
~SvpElementContainer()121*b1cdbd2cSJim Jagielski SvpElementContainer::~SvpElementContainer()
122*b1cdbd2cSJim Jagielski {
123*b1cdbd2cSJim Jagielski }
124*b1cdbd2cSJim Jagielski 
SvpContainerThread(void * pSvpContainer)125*b1cdbd2cSJim Jagielski void SAL_CALL SvpContainerThread(void* pSvpContainer)
126*b1cdbd2cSJim Jagielski {
127*b1cdbd2cSJim Jagielski     ((SvpElementContainer*)pSvpContainer)->run();
128*b1cdbd2cSJim Jagielski }
129*b1cdbd2cSJim Jagielski 
run()130*b1cdbd2cSJim Jagielski void SvpElementContainer::run()
131*b1cdbd2cSJim Jagielski {
132*b1cdbd2cSJim Jagielski     bool bRun = m_nSocket != 0;
133*b1cdbd2cSJim Jagielski     while( bRun )
134*b1cdbd2cSJim Jagielski     {
135*b1cdbd2cSJim Jagielski         int nLocalSocket = accept( m_nSocket, NULL, NULL );
136*b1cdbd2cSJim Jagielski         if( nLocalSocket < 0 )
137*b1cdbd2cSJim Jagielski         {
138*b1cdbd2cSJim Jagielski             bRun = false;
139*b1cdbd2cSJim Jagielski             perror( "accept() failed" );
140*b1cdbd2cSJim Jagielski         }
141*b1cdbd2cSJim Jagielski         else
142*b1cdbd2cSJim Jagielski         {
143*b1cdbd2cSJim Jagielski             Application::PostUserEvent( LINK( this, SvpElementContainer, processRequest ), (void*)nLocalSocket );
144*b1cdbd2cSJim Jagielski         }
145*b1cdbd2cSJim Jagielski     }
146*b1cdbd2cSJim Jagielski     if( m_nSocket )
147*b1cdbd2cSJim Jagielski         close( m_nSocket );
148*b1cdbd2cSJim Jagielski }
149*b1cdbd2cSJim Jagielski 
matchType(SvpElement * pEle)150*b1cdbd2cSJim Jagielski static const char* matchType( SvpElement* pEle )
151*b1cdbd2cSJim Jagielski {
152*b1cdbd2cSJim Jagielski     if( dynamic_cast<SvpSalBitmap*>(pEle) )
153*b1cdbd2cSJim Jagielski         return "Bitmap";
154*b1cdbd2cSJim Jagielski     else if( dynamic_cast<SvpSalFrame*>(pEle) )
155*b1cdbd2cSJim Jagielski         return "Frame";
156*b1cdbd2cSJim Jagielski     else if( dynamic_cast<SvpSalVirtualDevice*>(pEle) )
157*b1cdbd2cSJim Jagielski         return "VirtualDevice";
158*b1cdbd2cSJim Jagielski     return typeid(*pEle).name();
159*b1cdbd2cSJim Jagielski }
160*b1cdbd2cSJim Jagielski 
IMPL_LINK(SvpElementContainer,processRequest,void *,pSocket)161*b1cdbd2cSJim Jagielski IMPL_LINK( SvpElementContainer, processRequest, void*, pSocket )
162*b1cdbd2cSJim Jagielski {
163*b1cdbd2cSJim Jagielski     int nFile = (int)pSocket;
164*b1cdbd2cSJim Jagielski 
165*b1cdbd2cSJim Jagielski     rtl::OStringBuffer aBuf( 256 ), aAnswer( 256 );
166*b1cdbd2cSJim Jagielski     char c;
167*b1cdbd2cSJim Jagielski     while( read( nFile, &c, 1 ) && c != '\n' )
168*b1cdbd2cSJim Jagielski         aBuf.append( sal_Char(c) );
169*b1cdbd2cSJim Jagielski     rtl::OString aCommand( aBuf.makeStringAndClear() );
170*b1cdbd2cSJim Jagielski     if( aCommand.compareTo( "list", 4 ) == 0 )
171*b1cdbd2cSJim Jagielski     {
172*b1cdbd2cSJim Jagielski         std::hash_map< rtl::OString, std::list<SvpElement*>, rtl::OStringHash > aMap;
173*b1cdbd2cSJim Jagielski         for( std::list< SvpElement* >::const_iterator it = m_aElements.begin();
174*b1cdbd2cSJim Jagielski              it != m_aElements.end(); ++it )
175*b1cdbd2cSJim Jagielski         {
176*b1cdbd2cSJim Jagielski             std::list<SvpElement*>& rList = aMap[matchType(*it)];
177*b1cdbd2cSJim Jagielski             rList.push_back( *it );
178*b1cdbd2cSJim Jagielski         }
179*b1cdbd2cSJim Jagielski         for( std::hash_map< rtl::OString, std::list<SvpElement*>, rtl::OStringHash>::const_iterator hash_it = aMap.begin();
180*b1cdbd2cSJim Jagielski              hash_it != aMap.end(); ++hash_it )
181*b1cdbd2cSJim Jagielski         {
182*b1cdbd2cSJim Jagielski             aAnswer.append( "ElementType: " );
183*b1cdbd2cSJim Jagielski             aAnswer.append( hash_it->first );
184*b1cdbd2cSJim Jagielski             aAnswer.append( '\n' );
185*b1cdbd2cSJim Jagielski             for( std::list<SvpElement*>::const_iterator it = hash_it->second.begin();
186*b1cdbd2cSJim Jagielski                  it != hash_it->second.end(); ++it )
187*b1cdbd2cSJim Jagielski             {
188*b1cdbd2cSJim Jagielski                 aAnswer.append( sal_Int64(reinterpret_cast<sal_uInt32>(*it)), 16 );
189*b1cdbd2cSJim Jagielski                 aAnswer.append( '\n' );
190*b1cdbd2cSJim Jagielski             }
191*b1cdbd2cSJim Jagielski         }
192*b1cdbd2cSJim Jagielski     }
193*b1cdbd2cSJim Jagielski     else if( aCommand.compareTo( "get", 3 ) == 0 )
194*b1cdbd2cSJim Jagielski     {
195*b1cdbd2cSJim Jagielski         sal_IntPtr aId = aCommand.copy( 3 ).toInt64( 16 );
196*b1cdbd2cSJim Jagielski         SvpElement* pElement = reinterpret_cast<SvpElement*>(aId);
197*b1cdbd2cSJim Jagielski         for( std::list< SvpElement* >::const_iterator it = m_aElements.begin();
198*b1cdbd2cSJim Jagielski              it != m_aElements.end(); ++it )
199*b1cdbd2cSJim Jagielski         {
200*b1cdbd2cSJim Jagielski             if( *it == pElement )
201*b1cdbd2cSJim Jagielski             {
202*b1cdbd2cSJim Jagielski                 const basebmp::BitmapDeviceSharedPtr& rDevice = pElement->getDevice();
203*b1cdbd2cSJim Jagielski                 if( rDevice.get() )
204*b1cdbd2cSJim Jagielski                 {
205*b1cdbd2cSJim Jagielski                     SvpSalBitmap* pSalBitmap = new SvpSalBitmap();
206*b1cdbd2cSJim Jagielski                     pSalBitmap->setBitmap( rDevice );
207*b1cdbd2cSJim Jagielski                     Bitmap aBitmap( pSalBitmap );
208*b1cdbd2cSJim Jagielski                     SvMemoryStream aStream( 256, 256 );
209*b1cdbd2cSJim Jagielski                     aStream << aBitmap;
210*b1cdbd2cSJim Jagielski                     aStream.Seek( STREAM_SEEK_TO_END );
211*b1cdbd2cSJim Jagielski                     int nBytes = aStream.Tell();
212*b1cdbd2cSJim Jagielski                     aStream.Seek( STREAM_SEEK_TO_BEGIN );
213*b1cdbd2cSJim Jagielski                     aAnswer.append( (const sal_Char*)aStream.GetData(), nBytes );
214*b1cdbd2cSJim Jagielski                 }
215*b1cdbd2cSJim Jagielski                 break;
216*b1cdbd2cSJim Jagielski             }
217*b1cdbd2cSJim Jagielski         }
218*b1cdbd2cSJim Jagielski     }
219*b1cdbd2cSJim Jagielski     else if( aCommand.compareTo( "quit", 4 ) == 0 )
220*b1cdbd2cSJim Jagielski     {
221*b1cdbd2cSJim Jagielski         Application::Quit();
222*b1cdbd2cSJim Jagielski         close( m_nSocket );
223*b1cdbd2cSJim Jagielski         m_nSocket = 0;
224*b1cdbd2cSJim Jagielski     }
225*b1cdbd2cSJim Jagielski     write( nFile, aAnswer.getStr(), aAnswer.getLength() );
226*b1cdbd2cSJim Jagielski     close( nFile );
227*b1cdbd2cSJim Jagielski 
228*b1cdbd2cSJim Jagielski     return 0;
229*b1cdbd2cSJim Jagielski }
230*b1cdbd2cSJim Jagielski 
231*b1cdbd2cSJim Jagielski #endif
232*b1cdbd2cSJim Jagielski 
233*b1cdbd2cSJim Jagielski using namespace basebmp;
234*b1cdbd2cSJim Jagielski 
SvpElement()235*b1cdbd2cSJim Jagielski SvpElement::SvpElement()
236*b1cdbd2cSJim Jagielski {
237*b1cdbd2cSJim Jagielski     #if defined WITH_SVP_LISTENING
238*b1cdbd2cSJim Jagielski     SvpElementContainer::get().registerElement( this );
239*b1cdbd2cSJim Jagielski     #endif
240*b1cdbd2cSJim Jagielski }
241*b1cdbd2cSJim Jagielski 
~SvpElement()242*b1cdbd2cSJim Jagielski SvpElement::~SvpElement()
243*b1cdbd2cSJim Jagielski {
244*b1cdbd2cSJim Jagielski     #if defined WITH_SVP_LISTENING
245*b1cdbd2cSJim Jagielski     SvpElementContainer::get().deregisterElement( this );
246*b1cdbd2cSJim Jagielski     #endif
247*b1cdbd2cSJim Jagielski }
248*b1cdbd2cSJim Jagielski 
getBitCountFromScanlineFormat(sal_Int32 nFormat)249*b1cdbd2cSJim Jagielski sal_uInt32 SvpElement::getBitCountFromScanlineFormat( sal_Int32 nFormat )
250*b1cdbd2cSJim Jagielski {
251*b1cdbd2cSJim Jagielski     sal_uInt32 nBitCount = 1;
252*b1cdbd2cSJim Jagielski     switch( nFormat )
253*b1cdbd2cSJim Jagielski     {
254*b1cdbd2cSJim Jagielski         case Format::ONE_BIT_MSB_GREY:
255*b1cdbd2cSJim Jagielski         case Format::ONE_BIT_LSB_GREY:
256*b1cdbd2cSJim Jagielski         case Format::ONE_BIT_MSB_PAL:
257*b1cdbd2cSJim Jagielski         case Format::ONE_BIT_LSB_PAL:
258*b1cdbd2cSJim Jagielski             nBitCount = 1;
259*b1cdbd2cSJim Jagielski             break;
260*b1cdbd2cSJim Jagielski         case Format::FOUR_BIT_MSB_GREY:
261*b1cdbd2cSJim Jagielski         case Format::FOUR_BIT_LSB_GREY:
262*b1cdbd2cSJim Jagielski         case Format::FOUR_BIT_MSB_PAL:
263*b1cdbd2cSJim Jagielski         case Format::FOUR_BIT_LSB_PAL:
264*b1cdbd2cSJim Jagielski             nBitCount = 4;
265*b1cdbd2cSJim Jagielski             break;
266*b1cdbd2cSJim Jagielski         case Format::EIGHT_BIT_PAL:
267*b1cdbd2cSJim Jagielski         case Format::EIGHT_BIT_GREY:
268*b1cdbd2cSJim Jagielski             nBitCount = 8;
269*b1cdbd2cSJim Jagielski             break;
270*b1cdbd2cSJim Jagielski         case Format::SIXTEEN_BIT_LSB_TC_MASK:
271*b1cdbd2cSJim Jagielski         case Format::SIXTEEN_BIT_MSB_TC_MASK:
272*b1cdbd2cSJim Jagielski             nBitCount = 16;
273*b1cdbd2cSJim Jagielski             break;
274*b1cdbd2cSJim Jagielski         case Format::TWENTYFOUR_BIT_TC_MASK:
275*b1cdbd2cSJim Jagielski             nBitCount = 24;
276*b1cdbd2cSJim Jagielski             break;
277*b1cdbd2cSJim Jagielski         case Format::THIRTYTWO_BIT_TC_MASK:
278*b1cdbd2cSJim Jagielski             nBitCount = 32;
279*b1cdbd2cSJim Jagielski             break;
280*b1cdbd2cSJim Jagielski         default:
281*b1cdbd2cSJim Jagielski         DBG_ERROR( "unsupported basebmp format" );
282*b1cdbd2cSJim Jagielski         break;
283*b1cdbd2cSJim Jagielski     }
284*b1cdbd2cSJim Jagielski     return nBitCount;
285*b1cdbd2cSJim Jagielski }
286*b1cdbd2cSJim Jagielski 
287*b1cdbd2cSJim Jagielski 
288