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