Version 6.4.0.3, tag libreoffice-6.4.0.3
[LibreOffice.git] / basic / source / comp / codegen.cxx
blobd8bbdcfa18fc2669c1be4e99f8383bf2bc8f6c6a
1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
2 /*
3 * This file is part of the LibreOffice project.
5 * This Source Code Form is subject to the terms of the Mozilla Public
6 * License, v. 2.0. If a copy of the MPL was not distributed with this
7 * file, You can obtain one at http://mozilla.org/MPL/2.0/.
9 * This file incorporates work covered by the following license notice:
11 * Licensed to the Apache Software Foundation (ASF) under one or more
12 * contributor license agreements. See the NOTICE file distributed
13 * with this work for additional information regarding copyright
14 * ownership. The ASF licenses this file to you under the Apache
15 * License, Version 2.0 (the "License"); you may not use this file
16 * except in compliance with the License. You may obtain a copy of
17 * the License at http://www.apache.org/licenses/LICENSE-2.0 .
21 #include <basic/sberrors.hxx>
22 #include <basic/sbx.hxx>
23 #include <basic/sbmeth.hxx>
24 #include <basic/sbmod.hxx>
25 #include <image.hxx>
26 #include <codegen.hxx>
27 #include <parser.hxx>
28 #include <sbintern.hxx>
29 #include <cstddef>
30 #include <limits>
31 #include <algorithm>
32 #include <string_view>
33 #include <osl/diagnose.h>
34 #include <rtl/ustrbuf.hxx>
35 #include <com/sun/star/script/ModuleType.hpp>
37 // nInc is the increment size of the buffers
39 SbiCodeGen::SbiCodeGen( SbModule& r, SbiParser* p, short nInc )
40 : rMod( r ), aCode( p, nInc )
42 pParser = p;
43 bStmnt = false;
44 nLine = 0;
45 nCol = 0;
46 nForLevel = 0;
49 sal_uInt32 SbiCodeGen::GetPC() const
51 return aCode.GetSize();
54 // memorize the statement
56 void SbiCodeGen::Statement()
58 if( pParser->IsCodeCompleting() )
59 return;
61 bStmnt = true;
63 nLine = pParser->GetLine();
64 nCol = pParser->GetCol1();
66 // #29955 Store the information of the for-loop-layer
67 // in the upper Byte of the column
68 nCol = (nCol & 0xff) + 0x100 * nForLevel;
71 // Mark the beginning of a statement
73 void SbiCodeGen::GenStmnt()
75 if( pParser->IsCodeCompleting() )
76 return;
78 if( bStmnt )
80 bStmnt = false;
81 Gen( SbiOpcode::STMNT_, nLine, nCol );
85 // The Gen-Routines return the offset of the 1. operand,
86 // so that jumps can sink their backchain there.
88 sal_uInt32 SbiCodeGen::Gen( SbiOpcode eOpcode )
90 if( pParser->IsCodeCompleting() )
91 return 0;
93 #ifdef DBG_UTIL
94 if( eOpcode < SbiOpcode::SbOP0_START || eOpcode > SbiOpcode::SbOP0_END )
95 pParser->Error( ERRCODE_BASIC_INTERNAL_ERROR, "OPCODE1" );
96 #endif
97 GenStmnt();
98 aCode += static_cast<sal_uInt8>(eOpcode);
99 return GetPC();
102 sal_uInt32 SbiCodeGen::Gen( SbiOpcode eOpcode, sal_uInt32 nOpnd )
104 if( pParser->IsCodeCompleting() )
105 return 0;
107 #ifdef DBG_UTIL
108 if( eOpcode < SbiOpcode::SbOP1_START || eOpcode > SbiOpcode::SbOP1_END )
109 pParser->Error( ERRCODE_BASIC_INTERNAL_ERROR, "OPCODE2" );
110 #endif
111 GenStmnt();
112 aCode += static_cast<sal_uInt8>(eOpcode);
113 sal_uInt32 n = GetPC();
114 aCode += nOpnd;
115 return n;
118 sal_uInt32 SbiCodeGen::Gen( SbiOpcode eOpcode, sal_uInt32 nOpnd1, sal_uInt32 nOpnd2 )
120 if( pParser->IsCodeCompleting() )
121 return 0;
123 #ifdef DBG_UTIL
124 if( eOpcode < SbiOpcode::SbOP2_START || eOpcode > SbiOpcode::SbOP2_END )
125 pParser->Error( ERRCODE_BASIC_INTERNAL_ERROR, "OPCODE3" );
126 #endif
127 GenStmnt();
128 aCode += static_cast<sal_uInt8>(eOpcode);
129 sal_uInt32 n = GetPC();
130 aCode += nOpnd1;
131 aCode += nOpnd2;
132 return n;
135 // Storing of the created image in the module
137 void SbiCodeGen::Save()
139 if( pParser->IsCodeCompleting() )
140 return;
142 SbiImage* p = new SbiImage;
143 rMod.StartDefinitions();
144 // OPTION BASE-Value:
145 p->nDimBase = pParser->nBase;
146 // OPTION take over the EXPLICIT-Flag
147 if( pParser->bExplicit )
148 p->SetFlag( SbiImageFlags::EXPLICIT );
150 int nIfaceCount = 0;
151 if( rMod.mnType == css::script::ModuleType::CLASS )
153 rMod.bIsProxyModule = true;
154 p->SetFlag( SbiImageFlags::CLASSMODULE );
155 GetSbData()->pClassFac->AddClassModule( &rMod );
157 nIfaceCount = pParser->aIfaceVector.size();
158 if( !rMod.pClassData )
159 rMod.pClassData.reset(new SbClassData);
160 if( nIfaceCount )
162 for( int i = 0 ; i < nIfaceCount ; i++ )
164 const OUString& rIfaceName = pParser->aIfaceVector[i];
165 SbxVariable* pIfaceVar = new SbxVariable( SbxVARIANT );
166 pIfaceVar->SetName( rIfaceName );
167 SbxArray* pIfaces = rMod.pClassData->mxIfaces.get();
168 pIfaces->Insert( pIfaceVar, pIfaces->Count() );
172 rMod.pClassData->maRequiredTypes = pParser->aRequiredTypes;
174 else
176 GetSbData()->pClassFac->RemoveClassModule( &rMod );
177 // Only a ClassModule can revert to Normal
178 if ( rMod.mnType == css::script::ModuleType::CLASS )
180 rMod.mnType = css::script::ModuleType::NORMAL;
182 rMod.bIsProxyModule = false;
185 // GlobalCode-Flag
186 if( pParser->HasGlobalCode() )
188 p->SetFlag( SbiImageFlags::INITCODE );
190 // The entry points:
191 for( SbiSymDef* pDef = pParser->aPublics.First(); pDef;
192 pDef = pParser->aPublics.Next() )
194 SbiProcDef* pProc = pDef->GetProcDef();
195 if( pProc && pProc->IsDefined() )
197 OUString aProcName = pProc->GetName();
198 OUStringBuffer aIfaceProcName;
199 OUString aIfaceName;
200 sal_uInt16 nPassCount = 1;
201 if( nIfaceCount )
203 int nPropPrefixFound = aProcName.indexOf("Property ");
204 OUString aPureProcName = aProcName;
205 OUString aPropPrefix;
206 if( nPropPrefixFound == 0 )
208 aPropPrefix = aProcName.copy( 0, 13 ); // 13 == Len( "Property ?et " )
209 aPureProcName = aProcName.copy( 13 );
211 for( int i = 0 ; i < nIfaceCount ; i++ )
213 const OUString& rIfaceName = pParser->aIfaceVector[i];
214 int nFound = aPureProcName.indexOf( rIfaceName );
215 if( nFound == 0 && aPureProcName[rIfaceName.getLength()] == '_' )
217 if( nPropPrefixFound == 0 )
219 aIfaceProcName.append(aPropPrefix);
221 aIfaceProcName.append(std::u16string_view(aPureProcName).substr(rIfaceName.getLength() + 1) );
222 aIfaceName = rIfaceName;
223 nPassCount = 2;
224 break;
228 SbMethod* pMeth = nullptr;
229 for( sal_uInt16 nPass = 0 ; nPass < nPassCount ; nPass++ )
231 if( nPass == 1 )
233 aProcName = aIfaceProcName.toString();
235 PropertyMode ePropMode = pProc->getPropertyMode();
236 if( ePropMode != PropertyMode::NONE )
238 SbxDataType ePropType = SbxEMPTY;
239 switch( ePropMode )
241 case PropertyMode::Get:
242 ePropType = pProc->GetType();
243 break;
244 case PropertyMode::Let:
246 // type == type of first parameter
247 ePropType = SbxVARIANT; // Default
248 SbiSymPool* pPool = &pProc->GetParams();
249 if( pPool->GetSize() > 1 )
251 SbiSymDef* pPar = pPool->Get( 1 );
252 if( pPar )
254 ePropType = pPar->GetType();
257 break;
259 case PropertyMode::Set:
260 ePropType = SbxOBJECT;
261 break;
262 default:
263 OSL_FAIL("Illegal PropertyMode");
264 break;
266 OUString aPropName = pProc->GetPropName();
267 if( nPass == 1 )
269 aPropName = aPropName.copy( aIfaceName.getLength() + 1 );
271 rMod.GetProcedureProperty( aPropName, ePropType );
273 if( nPass == 1 )
275 rMod.GetIfaceMapperMethod( aProcName, pMeth );
277 else
279 pMeth = rMod.GetMethod( aProcName, pProc->GetType() );
281 if( !pProc->IsPublic() )
283 pMeth->SetFlag( SbxFlagBits::Private );
285 // Declare? -> Hidden
286 if( !pProc->GetLib().isEmpty())
288 pMeth->SetFlag( SbxFlagBits::Hidden );
290 pMeth->nStart = pProc->GetAddr();
291 pMeth->nLine1 = pProc->GetLine1();
292 pMeth->nLine2 = pProc->GetLine2();
293 // The parameter:
294 SbxInfo* pInfo = pMeth->GetInfo();
295 OUString aHelpFile, aComment;
296 sal_uInt32 nHelpId = 0;
297 if( pInfo )
299 // Rescue the additional data
300 aHelpFile = pInfo->GetHelpFile();
301 aComment = pInfo->GetComment();
302 nHelpId = pInfo->GetHelpId();
304 // And reestablish the parameter list
305 pInfo = new SbxInfo( aHelpFile, nHelpId );
306 pInfo->SetComment( aComment );
307 SbiSymPool* pPool = &pProc->GetParams();
308 // The first element is always the value of the function!
309 for( sal_uInt16 i = 1; i < pPool->GetSize(); i++ )
311 SbiSymDef* pPar = pPool->Get( i );
312 SbxDataType t = pPar->GetType();
313 if( !pPar->IsByVal() )
315 t = static_cast<SbxDataType>( t | SbxBYREF );
317 if( pPar->GetDims() )
319 t = static_cast<SbxDataType>( t | SbxARRAY );
321 // #33677 hand-over an Optional-Info
322 SbxFlagBits nFlags = SbxFlagBits::Read;
323 if( pPar->IsOptional() )
325 nFlags |= SbxFlagBits::Optional;
327 pInfo->AddParam( pPar->GetName(), t, nFlags );
329 sal_uInt32 nUserData = 0;
330 sal_uInt16 nDefaultId = pPar->GetDefaultId();
331 if( nDefaultId )
333 nUserData |= nDefaultId;
335 if( pPar->IsParamArray() )
337 nUserData |= PARAM_INFO_PARAMARRAY;
339 if( pPar->IsWithBrackets() )
341 nUserData |= PARAM_INFO_WITHBRACKETS;
343 SbxParamInfo* pParam = nullptr;
344 if( nUserData )
346 pParam = const_cast<SbxParamInfo*>(pInfo->GetParam( i ));
348 if( pParam )
350 pParam->nUserData = nUserData;
353 pMeth->SetInfo( pInfo );
355 } // for( iPass...
358 // The code
359 p->AddCode( std::unique_ptr<char[]>(aCode.GetBuffer()), aCode.GetSize() );
361 // The global StringPool. 0 is not occupied.
362 SbiStringPool* pPool = &pParser->aGblStrings;
363 sal_uInt16 nSize = pPool->GetSize();
364 p->MakeStrings( nSize );
365 sal_uInt16 i;
366 for( i = 1; i <= nSize; i++ )
368 p->AddString( pPool->Find( i ) );
370 // Insert types
371 sal_uInt16 nCount = pParser->rTypeArray->Count();
372 for (i = 0; i < nCount; i++)
374 p->AddType(static_cast<SbxObject *>(pParser->rTypeArray->Get(i)));
376 // Insert enum objects
377 nCount = pParser->rEnumArray->Count();
378 for (i = 0; i < nCount; i++)
380 p->AddEnum(static_cast<SbxObject *>(pParser->rEnumArray->Get(i)));
382 if( !p->IsError() )
384 rMod.pImage = p;
386 else
388 delete p;
390 rMod.EndDefinitions();
393 template < class T >
394 class PCodeVisitor
396 public:
397 virtual ~PCodeVisitor();
399 virtual void start( const sal_uInt8* pStart ) = 0;
400 virtual void processOpCode0( SbiOpcode eOp ) = 0;
401 virtual void processOpCode1( SbiOpcode eOp, T nOp1 ) = 0;
402 virtual void processOpCode2( SbiOpcode eOp, T nOp1, T nOp2 ) = 0;
403 virtual bool processParams() = 0;
406 template <class T> PCodeVisitor< T >::~PCodeVisitor()
409 template <class T>
410 class PCodeBufferWalker
412 private:
413 T m_nBytes;
414 const sal_uInt8* m_pCode;
415 static T readParam( sal_uInt8 const *& pCode )
417 T nOp1=0;
418 for ( std::size_t i=0; i<sizeof( T ); ++i )
419 nOp1 |= *pCode++ << ( i * 8);
420 return nOp1;
422 public:
423 PCodeBufferWalker( const sal_uInt8* pCode, T nBytes ): m_nBytes( nBytes ), m_pCode( pCode )
426 void visitBuffer( PCodeVisitor< T >& visitor )
428 const sal_uInt8* pCode = m_pCode;
429 if ( !pCode )
430 return;
431 const sal_uInt8* pEnd = pCode + m_nBytes;
432 visitor.start( m_pCode );
433 T nOp1 = 0, nOp2 = 0;
434 for( ; pCode < pEnd; )
436 SbiOpcode eOp = static_cast<SbiOpcode>(*pCode++);
438 if ( eOp <= SbiOpcode::SbOP0_END )
439 visitor.processOpCode0( eOp );
440 else if( eOp >= SbiOpcode::SbOP1_START && eOp <= SbiOpcode::SbOP1_END )
442 if ( visitor.processParams() )
443 nOp1 = readParam( pCode );
444 else
445 pCode += sizeof( T );
446 visitor.processOpCode1( eOp, nOp1 );
448 else if( eOp >= SbiOpcode::SbOP2_START && eOp <= SbiOpcode::SbOP2_END )
450 if ( visitor.processParams() )
452 nOp1 = readParam( pCode );
453 nOp2 = readParam( pCode );
455 else
456 pCode += ( sizeof( T ) * 2 );
457 visitor.processOpCode2( eOp, nOp1, nOp2 );
463 template < class T, class S >
464 class OffSetAccumulator : public PCodeVisitor< T >
466 T m_nNumOp0;
467 T m_nNumSingleParams;
468 T m_nNumDoubleParams;
469 public:
471 OffSetAccumulator() : m_nNumOp0(0), m_nNumSingleParams(0), m_nNumDoubleParams(0){}
472 virtual void start( const sal_uInt8* /*pStart*/ ) override {}
473 virtual void processOpCode0( SbiOpcode /*eOp*/ ) override { ++m_nNumOp0; }
474 virtual void processOpCode1( SbiOpcode /*eOp*/, T /*nOp1*/ ) override { ++m_nNumSingleParams; }
475 virtual void processOpCode2( SbiOpcode /*eOp*/, T /*nOp1*/, T /*nOp2*/ ) override { ++m_nNumDoubleParams; }
476 S offset()
478 typedef decltype(T(1) + S(1)) larger_t; // type capable to hold both value ranges of T and S
479 T result = 0 ;
480 static const S max = std::numeric_limits< S >::max();
481 result = m_nNumOp0 + ( ( sizeof(S) + 1 ) * m_nNumSingleParams ) + ( (( sizeof(S) * 2 )+ 1 ) * m_nNumDoubleParams );
482 return std::min<larger_t>(max, result);
484 virtual bool processParams() override { return false; }
488 template < class T, class S >
489 class BufferTransformer : public PCodeVisitor< T >
491 const sal_uInt8* m_pStart;
492 SbiBuffer m_ConvertedBuf;
493 public:
494 BufferTransformer():m_pStart(nullptr), m_ConvertedBuf( nullptr, 1024 ) {}
495 virtual void start( const sal_uInt8* pStart ) override { m_pStart = pStart; }
496 virtual void processOpCode0( SbiOpcode eOp ) override
498 m_ConvertedBuf += static_cast<sal_uInt8>(eOp);
500 virtual void processOpCode1( SbiOpcode eOp, T nOp1 ) override
502 m_ConvertedBuf += static_cast<sal_uInt8>(eOp);
503 switch( eOp )
505 case SbiOpcode::JUMP_:
506 case SbiOpcode::JUMPT_:
507 case SbiOpcode::JUMPF_:
508 case SbiOpcode::GOSUB_:
509 case SbiOpcode::CASEIS_:
510 case SbiOpcode::RETURN_:
511 case SbiOpcode::ERRHDL_:
512 case SbiOpcode::TESTFOR_:
513 nOp1 = static_cast<T>( convertBufferOffSet(m_pStart, nOp1) );
514 break;
515 case SbiOpcode::RESUME_:
516 if ( nOp1 > 1 )
517 nOp1 = static_cast<T>( convertBufferOffSet(m_pStart, nOp1) );
518 break;
519 default:
520 break;
523 m_ConvertedBuf += static_cast<S>(nOp1);
525 virtual void processOpCode2( SbiOpcode eOp, T nOp1, T nOp2 ) override
527 m_ConvertedBuf += static_cast<sal_uInt8>(eOp);
528 if ( eOp == SbiOpcode::CASEIS_ && nOp1 )
529 nOp1 = static_cast<T>( convertBufferOffSet(m_pStart, nOp1) );
530 m_ConvertedBuf += static_cast<S>(nOp1);
531 m_ConvertedBuf += static_cast<S>(nOp2);
534 virtual bool processParams() override { return true; }
535 // yeuch, careful here, you can only call
536 // GetBuffer on the returned SbiBuffer once, also
537 // you (as the caller) get to own the memory
538 SbiBuffer& buffer()
540 return m_ConvertedBuf;
542 static S convertBufferOffSet( const sal_uInt8* pStart, T nOp1 )
544 PCodeBufferWalker< T > aBuff( pStart, nOp1);
545 OffSetAccumulator< T, S > aVisitor;
546 aBuff.visitBuffer( aVisitor );
547 return aVisitor.offset();
551 sal_uInt32
552 SbiCodeGen::calcNewOffSet( sal_uInt8 const * pCode, sal_uInt16 nOffset )
554 return BufferTransformer< sal_uInt16, sal_uInt32 >::convertBufferOffSet( pCode, nOffset );
557 sal_uInt16
558 SbiCodeGen::calcLegacyOffSet( sal_uInt8 const * pCode, sal_uInt32 nOffset )
560 return BufferTransformer< sal_uInt32, sal_uInt16 >::convertBufferOffSet( pCode, nOffset );
563 template <class T, class S>
564 void
565 PCodeBuffConvertor<T,S>::convert()
567 PCodeBufferWalker< T > aBuf( m_pStart, m_nSize );
568 BufferTransformer< T, S > aTrnsfrmer;
569 aBuf.visitBuffer( aTrnsfrmer );
570 m_pCnvtdBuf = reinterpret_cast<sal_uInt8*>(aTrnsfrmer.buffer().GetBuffer());
571 m_nCnvtdSize = static_cast<S>( aTrnsfrmer.buffer().GetSize() );
574 template class PCodeBuffConvertor< sal_uInt16, sal_uInt32 >;
575 template class PCodeBuffConvertor< sal_uInt32, sal_uInt16 >;
577 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */