1 /************************************************************** 2 * 3 * Licensed to the Apache Software Foundation (ASF) under one 4 * or more contributor license agreements. See the NOTICE file 5 * distributed with this work for additional information 6 * regarding copyright ownership. The ASF licenses this file 7 * to you under the Apache License, Version 2.0 (the 8 * "License"); you may not use this file except in compliance 9 * with the License. You may obtain a copy of the License at 10 * 11 * http://www.apache.org/licenses/LICENSE-2.0 12 * 13 * Unless required by applicable law or agreed to in writing, 14 * software distributed under the License is distributed on an 15 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY 16 * KIND, either express or implied. See the License for the 17 * specific language governing permissions and limitations 18 * under the License. 19 * 20 *************************************************************/ 21 22 23 24 // MARKER(update_precomp.py): autogen include statement, do not remove 25 #include "precompiled_sc.hxx" 26 27 // INCLUDE --------------------------------------------------------------- 28 29 #ifndef INCLUDED_RTL_MATH_HXX 30 #include <rtl/math.hxx> 31 #endif 32 #include <rtl/logfile.hxx> 33 #include <string.h> 34 #include <math.h> 35 #include <stdio.h> 36 37 #if OSL_DEBUG_LEVEL > 1 38 #include <stdio.h> 39 #endif 40 #include <unotools/bootstrap.hxx> 41 #include <svl/zforlist.hxx> 42 43 #include "interpre.hxx" 44 #include "global.hxx" 45 #include "compiler.hxx" 46 #include "cell.hxx" 47 #include "document.hxx" 48 #include "dociter.hxx" 49 #include "scmatrix.hxx" 50 #include "globstr.hrc" 51 #include "cellkeytranslator.hxx" 52 #include "osversiondef.hxx" 53 54 #include <string.h> 55 #include <math.h> 56 #include <vector> 57 58 using ::std::vector; 59 using namespace formula; 60 61 const double fInvEpsilon = 1.0E-7; 62 63 // ----------------------------------------------------------------------- 64 struct MatrixAdd : public ::std::binary_function<double,double,double> 65 { 66 inline double operator() (const double& lhs, const double& rhs) const 67 { 68 return ::rtl::math::approxAdd( lhs,rhs); 69 } 70 }; 71 struct MatrixSub : public ::std::binary_function<double,double,double> 72 { 73 inline double operator() (const double& lhs, const double& rhs) const 74 { 75 return ::rtl::math::approxSub( lhs,rhs); 76 } 77 }; 78 struct MatrixMul : public ::std::binary_function<double,double,double> 79 { 80 inline double operator() (const double& lhs, const double& rhs) const 81 { 82 return lhs * rhs; 83 } 84 }; 85 struct MatrixDiv : public ::std::binary_function<double,double,double> 86 { 87 inline double operator() (const double& lhs, const double& rhs) const 88 { 89 return ScInterpreter::div( lhs,rhs); 90 } 91 }; 92 struct MatrixPow : public ::std::binary_function<double,double,double> 93 { 94 inline double operator() (const double& lhs, const double& rhs) const 95 { 96 return ::pow( lhs,rhs); 97 } 98 }; 99 100 double ScInterpreter::ScGetGCD(double fx, double fy) 101 { 102 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "sc", "er", "ScInterpreter::div" ); 103 // By ODFF definition GCD(0,a) => a. This is also vital for the code in 104 // ScGCD() to work correctly with a preset fy=0.0 105 if (fy == 0.0) 106 return fx; 107 else if (fx == 0.0) 108 return fy; 109 else 110 { 111 double fz = fmod(fx, fy); 112 while (fz > 0.0) 113 { 114 fx = fy; 115 fy = fz; 116 fz = fmod(fx, fy); 117 } 118 return fy; 119 } 120 } 121 122 void ScInterpreter::ScGCD() 123 { 124 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "sc", "er", "ScInterpreter::ScGCD" ); 125 short nParamCount = GetByte(); 126 if ( MustHaveParamCountMin( nParamCount, 1 ) ) 127 { 128 double fx, fy = 0.0; 129 ScRange aRange; 130 size_t nRefInList = 0; 131 while (!nGlobalError && nParamCount-- > 0) 132 { 133 switch (GetStackType()) 134 { 135 case svDouble : 136 case svString: 137 case svSingleRef: 138 { 139 fx = ::rtl::math::approxFloor( GetDouble()); 140 if (fx < 0.0) 141 { 142 PushIllegalArgument(); 143 return; 144 } 145 fy = ScGetGCD(fx, fy); 146 } 147 break; 148 case svDoubleRef : 149 case svRefList : 150 { 151 sal_uInt16 nErr = 0; 152 PopDoubleRef( aRange, nParamCount, nRefInList); 153 double nCellVal; 154 ScValueIterator aValIter(pDok, aRange, glSubTotal); 155 if (aValIter.GetFirst(nCellVal, nErr)) 156 { 157 do 158 { 159 fx = ::rtl::math::approxFloor( nCellVal); 160 if (fx < 0.0) 161 { 162 PushIllegalArgument(); 163 return; 164 } 165 fy = ScGetGCD(fx, fy); 166 } while (nErr == 0 && aValIter.GetNext(nCellVal, nErr)); 167 } 168 SetError(nErr); 169 } 170 break; 171 case svMatrix : 172 { 173 ScMatrixRef pMat = PopMatrix(); 174 if (pMat) 175 { 176 SCSIZE nC, nR; 177 pMat->GetDimensions(nC, nR); 178 if (nC == 0 || nR == 0) 179 SetError(errIllegalArgument); 180 else 181 { 182 SCSIZE nCount = nC * nR; 183 for ( SCSIZE j = 0; j < nCount; j++ ) 184 { 185 if (!pMat->IsValue(j)) 186 { 187 PushIllegalArgument(); 188 return; 189 } 190 fx = ::rtl::math::approxFloor( pMat->GetDouble(j)); 191 if (fx < 0.0) 192 { 193 PushIllegalArgument(); 194 return; 195 } 196 fy = ScGetGCD(fx, fy); 197 } 198 } 199 } 200 } 201 break; 202 default : SetError(errIllegalParameter); break; 203 } 204 } 205 PushDouble(fy); 206 } 207 } 208 209 void ScInterpreter:: ScLCM() 210 { 211 short nParamCount = GetByte(); 212 if ( MustHaveParamCountMin( nParamCount, 1 ) ) 213 { 214 double fx, fy = 1.0; 215 ScRange aRange; 216 size_t nRefInList = 0; 217 while (!nGlobalError && nParamCount-- > 0) 218 { 219 switch (GetStackType()) 220 { 221 case svDouble : 222 case svString: 223 case svSingleRef: 224 { 225 fx = ::rtl::math::approxFloor( GetDouble()); 226 if (fx < 0.0) 227 { 228 PushIllegalArgument(); 229 return; 230 } 231 if (fx == 0.0 || fy == 0.0) 232 fy = 0.0; 233 else 234 fy = fx * fy / ScGetGCD(fx, fy); 235 } 236 break; 237 case svDoubleRef : 238 case svRefList : 239 { 240 sal_uInt16 nErr = 0; 241 PopDoubleRef( aRange, nParamCount, nRefInList); 242 double nCellVal; 243 ScValueIterator aValIter(pDok, aRange, glSubTotal); 244 if (aValIter.GetFirst(nCellVal, nErr)) 245 { 246 do 247 { 248 fx = ::rtl::math::approxFloor( nCellVal); 249 if (fx < 0.0) 250 { 251 PushIllegalArgument(); 252 return; 253 } 254 if (fx == 0.0 || fy == 0.0) 255 fy = 0.0; 256 else 257 fy = fx * fy / ScGetGCD(fx, fy); 258 } while (nErr == 0 && aValIter.GetNext(nCellVal, nErr)); 259 } 260 SetError(nErr); 261 } 262 break; 263 case svMatrix : 264 { 265 ScMatrixRef pMat = PopMatrix(); 266 if (pMat) 267 { 268 SCSIZE nC, nR; 269 pMat->GetDimensions(nC, nR); 270 if (nC == 0 || nR == 0) 271 SetError(errIllegalArgument); 272 else 273 { 274 SCSIZE nCount = nC * nR; 275 for ( SCSIZE j = 0; j < nCount; j++ ) 276 { 277 if (!pMat->IsValue(j)) 278 { 279 PushIllegalArgument(); 280 return; 281 } 282 fx = ::rtl::math::approxFloor( pMat->GetDouble(j)); 283 if (fx < 0.0) 284 { 285 PushIllegalArgument(); 286 return; 287 } 288 if (fx == 0.0 || fy == 0.0) 289 fy = 0.0; 290 else 291 fy = fx * fy / ScGetGCD(fx, fy); 292 } 293 } 294 } 295 } 296 break; 297 default : SetError(errIllegalParameter); break; 298 } 299 } 300 PushDouble(fy); 301 } 302 } 303 304 ScMatrixRef ScInterpreter::GetNewMat(SCSIZE nC, SCSIZE nR) 305 { 306 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "sc", "er", "ScInterpreter::GetNewMat" ); 307 ScMatrix* pMat = new ScMatrix( nC, nR); 308 pMat->SetErrorInterpreter( this); 309 // A temporary matrix is mutable and ScMatrix::CloneIfConst() returns the 310 // very matrix. 311 pMat->SetImmutable( false); 312 SCSIZE nCols, nRows; 313 pMat->GetDimensions( nCols, nRows); 314 if ( nCols != nC || nRows != nR ) 315 { // arbitray limit of elements exceeded 316 SetError( errStackOverflow); 317 pMat->Delete(); 318 pMat = NULL; 319 } 320 return pMat; 321 } 322 323 ScMatrixRef ScInterpreter::CreateMatrixFromDoubleRef( const FormulaToken* pToken, 324 SCCOL nCol1, SCROW nRow1, SCTAB nTab1, 325 SCCOL nCol2, SCROW nRow2, SCTAB nTab2 ) 326 { 327 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "sc", "er", "ScInterpreter::CreateMatrixFromDoubleRef" ); 328 ScMatrixRef pMat = NULL; 329 if (nTab1 == nTab2 && !nGlobalError) 330 { 331 ScTokenMatrixMap::const_iterator aIter; 332 if ( static_cast<SCSIZE>(nRow2 - nRow1 + 1) * 333 static_cast<SCSIZE>(nCol2 - nCol1 + 1) > 334 ScMatrix::GetElementsMax() ) 335 SetError(errStackOverflow); 336 else if (pTokenMatrixMap && ((aIter = pTokenMatrixMap->find( pToken)) 337 != pTokenMatrixMap->end())) 338 pMat = static_cast<ScToken*>((*aIter).second.get())->GetMatrix(); 339 else 340 { 341 SCSIZE nMatCols = static_cast<SCSIZE>(nCol2 - nCol1 + 1); 342 SCSIZE nMatRows = static_cast<SCSIZE>(nRow2 - nRow1 + 1); 343 pMat = GetNewMat( nMatCols, nMatRows); 344 if (pMat && !nGlobalError) 345 { 346 // Set position where the next entry is expected. 347 SCROW nNextRow = nRow1; 348 SCCOL nNextCol = nCol1; 349 // Set last position as if there was a previous entry. 350 SCROW nThisRow = nRow2; 351 SCCOL nThisCol = nCol1 - 1; 352 ScCellIterator aCellIter( pDok, nCol1, nRow1, nTab1, nCol2, 353 nRow2, nTab2); 354 for (ScBaseCell* pCell = aCellIter.GetFirst(); pCell; pCell = 355 aCellIter.GetNext()) 356 { 357 nThisCol = aCellIter.GetCol(); 358 nThisRow = aCellIter.GetRow(); 359 if (nThisCol != nNextCol || nThisRow != nNextRow) 360 { 361 // Fill empty between iterator's positions. 362 for ( ; nNextCol <= nThisCol; ++nNextCol) 363 { 364 SCSIZE nC = nNextCol - nCol1; 365 SCSIZE nMatStopRow = ((nNextCol < nThisCol) ? 366 nMatRows : nThisRow - nRow1); 367 for (SCSIZE nR = nNextRow - nRow1; nR < 368 nMatStopRow; ++nR) 369 { 370 pMat->PutEmpty( nC, nR); 371 } 372 nNextRow = nRow1; 373 } 374 } 375 if (nThisRow == nRow2) 376 { 377 nNextCol = nThisCol + 1; 378 nNextRow = nRow1; 379 } 380 else 381 { 382 nNextCol = nThisCol; 383 nNextRow = nThisRow + 1; 384 } 385 if (HasCellEmptyData(pCell)) 386 { 387 pMat->PutEmpty( static_cast<SCSIZE>(nThisCol-nCol1), 388 static_cast<SCSIZE>(nThisRow-nRow1)); 389 } 390 else if (HasCellValueData(pCell)) 391 { 392 ScAddress aAdr( nThisCol, nThisRow, nTab1); 393 double fVal = GetCellValue( aAdr, pCell); 394 if ( nGlobalError ) 395 { 396 fVal = CreateDoubleError( nGlobalError); 397 nGlobalError = 0; 398 } 399 pMat->PutDouble( fVal, 400 static_cast<SCSIZE>(nThisCol-nCol1), 401 static_cast<SCSIZE>(nThisRow-nRow1)); 402 } 403 else 404 { 405 String aStr; 406 GetCellString( aStr, pCell); 407 if ( nGlobalError ) 408 { 409 double fVal = CreateDoubleError( nGlobalError); 410 nGlobalError = 0; 411 pMat->PutDouble( fVal, 412 static_cast<SCSIZE>(nThisCol-nCol1), 413 static_cast<SCSIZE>(nThisRow-nRow1)); 414 } 415 else 416 pMat->PutString( aStr, 417 static_cast<SCSIZE>(nThisCol-nCol1), 418 static_cast<SCSIZE>(nThisRow-nRow1)); 419 } 420 } 421 // Fill empty if iterator's last position wasn't the end. 422 if (nThisCol != nCol2 || nThisRow != nRow2) 423 { 424 for ( ; nNextCol <= nCol2; ++nNextCol) 425 { 426 SCSIZE nC = nNextCol - nCol1; 427 for (SCSIZE nR = nNextRow - nRow1; nR < nMatRows; ++nR) 428 { 429 pMat->PutEmpty( nC, nR); 430 } 431 nNextRow = nRow1; 432 } 433 } 434 if (pTokenMatrixMap) 435 pTokenMatrixMap->insert( ScTokenMatrixMap::value_type( 436 pToken, new ScMatrixToken( pMat))); 437 } 438 } 439 } 440 else // not a 2D matrix 441 SetError(errIllegalParameter); 442 return pMat; 443 } 444 445 446 ScMatrixRef ScInterpreter::GetMatrix() 447 { 448 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "sc", "er", "ScInterpreter::GetMatrix" ); 449 ScMatrixRef pMat = NULL; 450 switch (GetRawStackType()) 451 { 452 case svSingleRef : 453 { 454 ScAddress aAdr; 455 PopSingleRef( aAdr ); 456 pMat = GetNewMat(1, 1); 457 if (pMat) 458 { 459 ScBaseCell* pCell = GetCell( aAdr ); 460 if (HasCellEmptyData(pCell)) 461 pMat->PutEmpty( 0 ); 462 else if (HasCellValueData(pCell)) 463 pMat->PutDouble(GetCellValue(aAdr, pCell), 0); 464 else 465 { 466 String aStr; 467 GetCellString(aStr, pCell); 468 pMat->PutString(aStr, 0); 469 } 470 } 471 } 472 break; 473 case svDoubleRef: 474 { 475 SCCOL nCol1, nCol2; 476 SCROW nRow1, nRow2; 477 SCTAB nTab1, nTab2; 478 const ScToken* p = sp ? static_cast<const ScToken*>(pStack[sp-1]) : NULL; 479 PopDoubleRef(nCol1, nRow1, nTab1, nCol2, nRow2, nTab2); 480 pMat = CreateMatrixFromDoubleRef( p, nCol1, nRow1, nTab1, 481 nCol2, nRow2, nTab2); 482 } 483 break; 484 case svMatrix: 485 pMat = PopMatrix(); 486 break; 487 case svError : 488 case svMissing : 489 case svDouble : 490 { 491 double fVal = GetDouble(); 492 pMat = GetNewMat( 1, 1); 493 if ( pMat ) 494 { 495 if ( nGlobalError ) 496 { 497 fVal = CreateDoubleError( nGlobalError); 498 nGlobalError = 0; 499 } 500 pMat->PutDouble( fVal, 0); 501 } 502 } 503 break; 504 case svString : 505 { 506 String aStr = GetString(); 507 pMat = GetNewMat( 1, 1); 508 if ( pMat ) 509 { 510 if ( nGlobalError ) 511 { 512 double fVal = CreateDoubleError( nGlobalError); 513 pMat->PutDouble( fVal, 0); 514 nGlobalError = 0; 515 } 516 else 517 pMat->PutString( aStr, 0); 518 } 519 } 520 break; 521 default: 522 PopError(); 523 SetError( errIllegalArgument); 524 break; 525 } 526 return pMat; 527 } 528 529 void ScInterpreter::ScMatValue() 530 { 531 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "sc", "er", "ScInterpreter::ScMatValue" ); 532 if ( MustHaveParamCount( GetByte(), 3 ) ) 533 { 534 // 0 to count-1 535 SCSIZE nR = static_cast<SCSIZE>(::rtl::math::approxFloor(GetDouble())); 536 SCSIZE nC = static_cast<SCSIZE>(::rtl::math::approxFloor(GetDouble())); 537 switch (GetStackType()) 538 { 539 case svSingleRef : 540 { 541 ScAddress aAdr; 542 PopSingleRef( aAdr ); 543 ScBaseCell* pCell = GetCell( aAdr ); 544 if (pCell && pCell->GetCellType() == CELLTYPE_FORMULA) 545 { 546 sal_uInt16 nErrCode = ((ScFormulaCell*)pCell)->GetErrCode(); 547 if (nErrCode != 0) 548 PushError( nErrCode); 549 else 550 { 551 const ScMatrix* pMat = ((ScFormulaCell*)pCell)->GetMatrix(); 552 CalculateMatrixValue(pMat,nC,nR); 553 } 554 } 555 else 556 PushIllegalParameter(); 557 } 558 break; 559 case svDoubleRef : 560 { 561 SCCOL nCol1; 562 SCROW nRow1; 563 SCTAB nTab1; 564 SCCOL nCol2; 565 SCROW nRow2; 566 SCTAB nTab2; 567 PopDoubleRef(nCol1, nRow1, nTab1, nCol2, nRow2, nTab2); 568 if (nCol2 - nCol1 >= static_cast<SCCOL>(nR) && 569 nRow2 - nRow1 >= static_cast<SCROW>(nC) && 570 nTab1 == nTab2) 571 { 572 ScAddress aAdr( sal::static_int_cast<SCCOL>( nCol1 + nR ), 573 sal::static_int_cast<SCROW>( nRow1 + nC ), nTab1 ); 574 ScBaseCell* pCell = GetCell( aAdr ); 575 if (HasCellValueData(pCell)) 576 PushDouble(GetCellValue( aAdr, pCell )); 577 else 578 { 579 String aStr; 580 GetCellString(aStr, pCell); 581 PushString(aStr); 582 } 583 } 584 else 585 PushNoValue(); 586 } 587 break; 588 case svMatrix: 589 { 590 ScMatrixRef pMat = PopMatrix(); 591 CalculateMatrixValue(pMat,nC,nR); 592 } 593 break; 594 default: 595 PopError(); 596 PushIllegalParameter(); 597 break; 598 } 599 } 600 } 601 void ScInterpreter::CalculateMatrixValue(const ScMatrix* pMat,SCSIZE nC,SCSIZE nR) 602 { 603 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "sc", "er", "ScInterpreter::CalculateMatrixValue" ); 604 if (pMat) 605 { 606 SCSIZE nCl, nRw; 607 pMat->GetDimensions(nCl, nRw); 608 if (nC < nCl && nR < nRw) 609 { 610 ScMatValType nMatValType; 611 const ScMatrixValue* pMatVal = pMat->Get( nC, nR,nMatValType); 612 if (ScMatrix::IsNonValueType( nMatValType)) 613 PushString( pMatVal->GetString() ); 614 else 615 PushDouble(pMatVal->fVal); 616 // also handles DoubleError 617 } 618 else 619 PushNoValue(); 620 } 621 else 622 PushNoValue(); 623 } 624 625 void ScInterpreter::ScEMat() 626 { 627 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "sc", "er", "ScInterpreter::ScEMat" ); 628 if ( MustHaveParamCount( GetByte(), 1 ) ) 629 { 630 SCSIZE nDim = static_cast<SCSIZE>(::rtl::math::approxFloor(GetDouble())); 631 if ( nDim * nDim > ScMatrix::GetElementsMax() || nDim == 0) 632 PushIllegalArgument(); 633 else 634 { 635 ScMatrixRef pRMat = GetNewMat(nDim, nDim); 636 if (pRMat) 637 { 638 MEMat(pRMat, nDim); 639 PushMatrix(pRMat); 640 } 641 else 642 PushIllegalArgument(); 643 } 644 } 645 } 646 647 void ScInterpreter::MEMat(ScMatrix* mM, SCSIZE n) 648 { 649 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "sc", "er", "ScInterpreter::MEMat" ); 650 mM->FillDouble(0.0, 0, 0, n-1, n-1); 651 for (SCSIZE i = 0; i < n; i++) 652 mM->PutDouble(1.0, i, i); 653 } 654 655 /* Matrix LUP decomposition according to the pseudocode of "Introduction to 656 * Algorithms" by Cormen, Leiserson, Rivest, Stein. 657 * 658 * Added scaling for numeric stability. 659 * 660 * Given an n x n nonsingular matrix A, find a permutation matrix P, a unit 661 * lower-triangular matrix L, and an upper-triangular matrix U such that PA=LU. 662 * Compute L and U "in place" in the matrix A, the original content is 663 * destroyed. Note that the diagonal elements of the U triangular matrix 664 * replace the diagonal elements of the L-unit matrix (that are each ==1). The 665 * permutation matrix P is an array, where P[i]=j means that the i-th row of P 666 * contains a 1 in column j. Additionally keep track of the number of 667 * permutations (row exchanges). 668 * 669 * Returns 0 if a singular matrix is encountered, else +1 if an even number of 670 * permutations occured, or -1 if odd, which is the sign of the determinant. 671 * This may be used to calculate the determinant by multiplying the sign with 672 * the product of the diagonal elements of the LU matrix. 673 */ 674 static int lcl_LUP_decompose( ScMatrix* mA, const SCSIZE n, 675 ::std::vector< SCSIZE> & P ) 676 { 677 int nSign = 1; 678 // Find scale of each row. 679 ::std::vector< double> aScale(n); 680 for (SCSIZE i=0; i < n; ++i) 681 { 682 double fMax = 0.0; 683 for (SCSIZE j=0; j < n; ++j) 684 { 685 double fTmp = fabs( mA->GetDouble( j, i)); 686 if (fMax < fTmp) 687 fMax = fTmp; 688 } 689 if (fMax == 0.0) 690 return 0; // singular matrix 691 aScale[i] = 1.0 / fMax; 692 } 693 // Represent identity permutation, P[i]=i 694 for (SCSIZE i=0; i < n; ++i) 695 P[i] = i; 696 // "Recursion" on the diagonale. 697 SCSIZE l = n - 1; 698 for (SCSIZE k=0; k < l; ++k) 699 { 700 // Implicit pivoting. With the scale found for a row, compare values of 701 // a column and pick largest. 702 double fMax = 0.0; 703 double fScale = aScale[k]; 704 SCSIZE kp = k; 705 for (SCSIZE i = k; i < n; ++i) 706 { 707 double fTmp = fScale * fabs( mA->GetDouble( k, i)); 708 if (fMax < fTmp) 709 { 710 fMax = fTmp; 711 kp = i; 712 } 713 } 714 if (fMax == 0.0) 715 return 0; // singular matrix 716 // Swap rows. The pivot element will be at mA[k,kp] (row,col notation) 717 if (k != kp) 718 { 719 // permutations 720 SCSIZE nTmp = P[k]; 721 P[k] = P[kp]; 722 P[kp] = nTmp; 723 nSign = -nSign; 724 // scales 725 double fTmp = aScale[k]; 726 aScale[k] = aScale[kp]; 727 aScale[kp] = fTmp; 728 // elements 729 for (SCSIZE i=0; i < n; ++i) 730 { 731 double fMatTmp = mA->GetDouble( i, k); 732 mA->PutDouble( mA->GetDouble( i, kp), i, k); 733 mA->PutDouble( fMatTmp, i, kp); 734 } 735 } 736 // Compute Schur complement. 737 for (SCSIZE i = k+1; i < n; ++i) 738 { 739 double fTmp = mA->GetDouble( k, i) / mA->GetDouble( k, k); 740 mA->PutDouble( fTmp, k, i); 741 for (SCSIZE j = k+1; j < n; ++j) 742 mA->PutDouble( mA->GetDouble( j, i) - fTmp * mA->GetDouble( j, 743 k), j, i); 744 } 745 } 746 #if OSL_DEBUG_LEVEL > 1 747 fprintf( stderr, "\n%s\n", "lcl_LUP_decompose(): LU"); 748 for (SCSIZE i=0; i < n; ++i) 749 { 750 for (SCSIZE j=0; j < n; ++j) 751 fprintf( stderr, "%8.2g ", mA->GetDouble( j, i)); 752 fprintf( stderr, "\n%s\n", ""); 753 } 754 fprintf( stderr, "\n%s\n", "lcl_LUP_decompose(): P"); 755 for (SCSIZE j=0; j < n; ++j) 756 fprintf( stderr, "%5u ", (unsigned)P[j]); 757 fprintf( stderr, "\n%s\n", ""); 758 #endif 759 760 bool bSingular=false; 761 for (SCSIZE i=0; i < n && !bSingular; i++) 762 bSingular = (mA->GetDouble(i,i) == 0.0); 763 if (bSingular) 764 nSign = 0; 765 766 return nSign; 767 } 768 769 770 /* Solve a LUP decomposed equation Ax=b. LU is a combined matrix of L and U 771 * triangulars and P the permutation vector as obtained from 772 * lcl_LUP_decompose(). B is the right-hand side input vector, X is used to 773 * return the solution vector. 774 */ 775 static void lcl_LUP_solve( const ScMatrix* mLU, const SCSIZE n, 776 const ::std::vector< SCSIZE> & P, const ::std::vector< double> & B, 777 ::std::vector< double> & X ) 778 { 779 SCSIZE nFirst = SCSIZE_MAX; 780 // Ax=b => PAx=Pb, with decomposition LUx=Pb. 781 // Define y=Ux and solve for y in Ly=Pb using forward substitution. 782 for (SCSIZE i=0; i < n; ++i) 783 { 784 double fSum = B[P[i]]; 785 // Matrix inversion comes with a lot of zeros in the B vectors, we 786 // don't have to do all the computing with results multiplied by zero. 787 // Until then, simply lookout for the position of the first nonzero 788 // value. 789 if (nFirst != SCSIZE_MAX) 790 { 791 for (SCSIZE j = nFirst; j < i; ++j) 792 fSum -= mLU->GetDouble( j, i) * X[j]; // X[j] === y[j] 793 } 794 else if (fSum) 795 nFirst = i; 796 X[i] = fSum; // X[i] === y[i] 797 } 798 // Solve for x in Ux=y using back substitution. 799 for (SCSIZE i = n; i--; ) 800 { 801 double fSum = X[i]; // X[i] === y[i] 802 for (SCSIZE j = i+1; j < n; ++j) 803 fSum -= mLU->GetDouble( j, i) * X[j]; // X[j] === x[j] 804 X[i] = fSum / mLU->GetDouble( i, i); // X[i] === x[i] 805 } 806 #if OSL_DEBUG_LEVEL >1 807 fprintf( stderr, "\n%s\n", "lcl_LUP_solve():"); 808 for (SCSIZE i=0; i < n; ++i) 809 fprintf( stderr, "%8.2g ", X[i]); 810 fprintf( stderr, "%s\n", ""); 811 #endif 812 } 813 814 815 void ScInterpreter::ScMatDet() 816 { 817 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "sc", "er", "ScInterpreter::ScMatDet" ); 818 if ( MustHaveParamCount( GetByte(), 1 ) ) 819 { 820 ScMatrixRef pMat = GetMatrix(); 821 if (!pMat) 822 { 823 PushIllegalParameter(); 824 return; 825 } 826 if ( !pMat->IsNumeric() ) 827 { 828 PushNoValue(); 829 return; 830 } 831 SCSIZE nC, nR; 832 pMat->GetDimensions(nC, nR); 833 if ( nC != nR || nC == 0 || (sal_uLong) nC * nC > ScMatrix::GetElementsMax() ) 834 PushIllegalArgument(); 835 else 836 { 837 // LUP decomposition is done inplace, use copy. 838 ScMatrixRef xLU = pMat->Clone(); 839 if (!xLU) 840 PushError( errCodeOverflow); 841 else 842 { 843 ::std::vector< SCSIZE> P(nR); 844 int nDetSign = lcl_LUP_decompose( xLU, nR, P); 845 if (!nDetSign) 846 PushInt(0); // singular matrix 847 else 848 { 849 // In an LU matrix the determinant is simply the product of 850 // all diagonal elements. 851 double fDet = nDetSign; 852 ScMatrix* pLU = xLU; 853 for (SCSIZE i=0; i < nR; ++i) 854 fDet *= pLU->GetDouble( i, i); 855 PushDouble( fDet); 856 } 857 } 858 } 859 } 860 } 861 862 void ScInterpreter::ScMatInv() 863 { 864 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "sc", "er", "ScInterpreter::ScMatInv" ); 865 if ( MustHaveParamCount( GetByte(), 1 ) ) 866 { 867 ScMatrixRef pMat = GetMatrix(); 868 if (!pMat) 869 { 870 PushIllegalParameter(); 871 return; 872 } 873 if ( !pMat->IsNumeric() ) 874 { 875 PushNoValue(); 876 return; 877 } 878 SCSIZE nC, nR; 879 pMat->GetDimensions(nC, nR); 880 if ( nC != nR || nC == 0 || (sal_uLong) nC * nC > ScMatrix::GetElementsMax() ) 881 PushIllegalArgument(); 882 else 883 { 884 // LUP decomposition is done inplace, use copy. 885 ScMatrixRef xLU = pMat->Clone(); 886 // The result matrix. 887 ScMatrixRef xY = GetNewMat( nR, nR); 888 if (!xLU || !xY) 889 PushError( errCodeOverflow); 890 else 891 { 892 ::std::vector< SCSIZE> P(nR); 893 int nDetSign = lcl_LUP_decompose( xLU, nR, P); 894 if (!nDetSign) 895 PushIllegalArgument(); 896 else 897 { 898 // Solve equation for each column. 899 ScMatrix* pY = xY; 900 ::std::vector< double> B(nR); 901 ::std::vector< double> X(nR); 902 for (SCSIZE j=0; j < nR; ++j) 903 { 904 for (SCSIZE i=0; i < nR; ++i) 905 B[i] = 0.0; 906 B[j] = 1.0; 907 lcl_LUP_solve( xLU, nR, P, B, X); 908 for (SCSIZE i=0; i < nR; ++i) 909 pY->PutDouble( X[i], j, i); 910 } 911 #if 0 912 /* Possible checks for ill-condition: 913 * 1. Scale matrix, invert scaled matrix. If there are 914 * elements of the inverted matrix that are several 915 * orders of magnitude greater than 1 => 916 * ill-conditioned. 917 * Just how much is "several orders"? 918 * 2. Invert the inverted matrix and assess whether the 919 * result is sufficiently close to the original matrix. 920 * If not => ill-conditioned. 921 * Just what is sufficient? 922 * 3. Multiplying the inverse by the original matrix should 923 * produce a result sufficiently close to the identity 924 * matrix. 925 * Just what is sufficient? 926 * 927 * The following is #3. 928 */ 929 ScMatrixRef xR = GetNewMat( nR, nR); 930 if (xR) 931 { 932 ScMatrix* pR = xR; 933 lcl_MFastMult( pMat, pY, pR, nR, nR, nR); 934 #if OSL_DEBUG_LEVEL > 1 935 fprintf( stderr, "\n%s\n", "ScMatInv(): mult-identity"); 936 #endif 937 for (SCSIZE i=0; i < nR; ++i) 938 { 939 for (SCSIZE j=0; j < nR; ++j) 940 { 941 double fTmp = pR->GetDouble( j, i); 942 #if OSL_DEBUG_LEVEL > 1 943 fprintf( stderr, "%8.2g ", fTmp); 944 #endif 945 if (fabs( fTmp - (i == j)) > fInvEpsilon) 946 SetError( errIllegalArgument); 947 } 948 #if OSL_DEBUG_LEVEL > 1 949 fprintf( stderr, "\n%s\n", ""); 950 #endif 951 } 952 } 953 #endif 954 if (nGlobalError) 955 PushError( nGlobalError); 956 else 957 PushMatrix( pY); 958 } 959 } 960 } 961 } 962 } 963 964 void ScInterpreter::ScMatMult() 965 { 966 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "sc", "er", "ScInterpreter::ScMatMult" ); 967 if ( MustHaveParamCount( GetByte(), 2 ) ) 968 { 969 ScMatrixRef pMat2 = GetMatrix(); 970 ScMatrixRef pMat1 = GetMatrix(); 971 ScMatrixRef pRMat; 972 if (pMat1 && pMat2) 973 { 974 if ( pMat1->IsNumeric() && pMat2->IsNumeric() ) 975 { 976 SCSIZE nC1, nC2; 977 SCSIZE nR1, nR2; 978 pMat1->GetDimensions(nC1, nR1); 979 pMat2->GetDimensions(nC2, nR2); 980 if (nC1 != nR2) 981 PushIllegalArgument(); 982 else 983 { 984 pRMat = GetNewMat(nC2, nR1); 985 if (pRMat) 986 { 987 double sum; 988 for (SCSIZE i = 0; i < nR1; i++) 989 { 990 for (SCSIZE j = 0; j < nC2; j++) 991 { 992 sum = 0.0; 993 for (SCSIZE k = 0; k < nC1; k++) 994 { 995 sum += pMat1->GetDouble(k,i)*pMat2->GetDouble(j,k); 996 } 997 pRMat->PutDouble(sum, j, i); 998 } 999 } 1000 PushMatrix(pRMat); 1001 } 1002 else 1003 PushIllegalArgument(); 1004 } 1005 } 1006 else 1007 PushNoValue(); 1008 } 1009 else 1010 PushIllegalParameter(); 1011 } 1012 } 1013 1014 void ScInterpreter::ScMatTrans() 1015 { 1016 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "sc", "er", "ScInterpreter::ScMatTrans" ); 1017 if ( MustHaveParamCount( GetByte(), 1 ) ) 1018 { 1019 ScMatrixRef pMat = GetMatrix(); 1020 ScMatrixRef pRMat; 1021 if (pMat) 1022 { 1023 SCSIZE nC, nR; 1024 pMat->GetDimensions(nC, nR); 1025 pRMat = GetNewMat(nR, nC); 1026 if ( pRMat ) 1027 { 1028 pMat->MatTrans(*pRMat); 1029 PushMatrix(pRMat); 1030 } 1031 else 1032 PushIllegalArgument(); 1033 } 1034 else 1035 PushIllegalParameter(); 1036 } 1037 } 1038 1039 1040 /** Minimum extent of one result matrix dimension. 1041 For a row or column vector to be replicated the larger matrix dimension is 1042 returned, else the smaller dimension. 1043 */ 1044 inline SCSIZE lcl_GetMinExtent( SCSIZE n1, SCSIZE n2 ) 1045 { 1046 if (n1 == 1) 1047 return n2; 1048 else if (n2 == 1) 1049 return n1; 1050 else if (n1 < n2) 1051 return n1; 1052 else 1053 return n2; 1054 } 1055 1056 template<class _Function> 1057 ScMatrixRef lcl_MatrixCalculation(const _Function& _pOperation,ScMatrix* pMat1, ScMatrix* pMat2,ScInterpreter* _pIterpreter) 1058 { 1059 SCSIZE nC1, nC2, nMinC; 1060 SCSIZE nR1, nR2, nMinR; 1061 SCSIZE i, j; 1062 pMat1->GetDimensions(nC1, nR1); 1063 pMat2->GetDimensions(nC2, nR2); 1064 nMinC = lcl_GetMinExtent( nC1, nC2); 1065 nMinR = lcl_GetMinExtent( nR1, nR2); 1066 ScMatrixRef xResMat = _pIterpreter->GetNewMat(nMinC, nMinR); 1067 if (xResMat) 1068 { 1069 ScMatrix* pResMat = xResMat; 1070 for (i = 0; i < nMinC; i++) 1071 { 1072 for (j = 0; j < nMinR; j++) 1073 { 1074 if (pMat1->IsValueOrEmpty(i,j) && pMat2->IsValueOrEmpty(i,j)) 1075 { 1076 double d = _pOperation(pMat1->GetDouble(i,j),pMat2->GetDouble(i,j)); 1077 pResMat->PutDouble( d, i, j); 1078 } 1079 else 1080 pResMat->PutString(ScGlobal::GetRscString(STR_NO_VALUE), i, j); 1081 } 1082 } 1083 } 1084 return xResMat; 1085 } 1086 1087 ScMatrixRef ScInterpreter::MatConcat(ScMatrix* pMat1, ScMatrix* pMat2) 1088 { 1089 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "sc", "er", "ScInterpreter::MatConcat" ); 1090 SCSIZE nC1, nC2, nMinC; 1091 SCSIZE nR1, nR2, nMinR; 1092 SCSIZE i, j; 1093 pMat1->GetDimensions(nC1, nR1); 1094 pMat2->GetDimensions(nC2, nR2); 1095 nMinC = lcl_GetMinExtent( nC1, nC2); 1096 nMinR = lcl_GetMinExtent( nR1, nR2); 1097 ScMatrixRef xResMat = GetNewMat(nMinC, nMinR); 1098 if (xResMat) 1099 { 1100 ScMatrix* pResMat = xResMat; 1101 for (i = 0; i < nMinC; i++) 1102 { 1103 for (j = 0; j < nMinR; j++) 1104 { 1105 sal_uInt16 nErr = pMat1->GetErrorIfNotString( i, j); 1106 if (!nErr) 1107 nErr = pMat2->GetErrorIfNotString( i, j); 1108 if (nErr) 1109 pResMat->PutError( nErr, i, j); 1110 else 1111 { 1112 String aTmp( pMat1->GetString( *pFormatter, i, j)); 1113 aTmp += pMat2->GetString( *pFormatter, i, j); 1114 pResMat->PutString( aTmp, i, j); 1115 } 1116 } 1117 } 1118 } 1119 return xResMat; 1120 } 1121 1122 1123 // fuer DATE, TIME, DATETIME 1124 void lcl_GetDiffDateTimeFmtType( short& nFuncFmt, short nFmt1, short nFmt2 ) 1125 { 1126 if ( nFmt1 != NUMBERFORMAT_UNDEFINED || nFmt2 != NUMBERFORMAT_UNDEFINED ) 1127 { 1128 if ( nFmt1 == nFmt2 ) 1129 { 1130 if ( nFmt1 == NUMBERFORMAT_TIME || nFmt1 == NUMBERFORMAT_DATETIME ) 1131 nFuncFmt = NUMBERFORMAT_TIME; // Zeiten ergeben Zeit 1132 // else: nichts besonderes, Zahl (Datum - Datum := Tage) 1133 } 1134 else if ( nFmt1 == NUMBERFORMAT_UNDEFINED ) 1135 nFuncFmt = nFmt2; // z.B. Datum + Tage := Datum 1136 else if ( nFmt2 == NUMBERFORMAT_UNDEFINED ) 1137 nFuncFmt = nFmt1; 1138 else 1139 { 1140 if ( nFmt1 == NUMBERFORMAT_DATE || nFmt2 == NUMBERFORMAT_DATE || 1141 nFmt1 == NUMBERFORMAT_DATETIME || nFmt2 == NUMBERFORMAT_DATETIME ) 1142 { 1143 if ( nFmt1 == NUMBERFORMAT_TIME || nFmt2 == NUMBERFORMAT_TIME ) 1144 nFuncFmt = NUMBERFORMAT_DATETIME; // Datum + Zeit 1145 } 1146 } 1147 } 1148 } 1149 1150 1151 void ScInterpreter::ScAdd() 1152 { 1153 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "sc", "er", "ScInterpreter::ScAdd" ); 1154 CalculateAddSub(sal_False); 1155 } 1156 void ScInterpreter::CalculateAddSub(sal_Bool _bSub) 1157 { 1158 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "sc", "er", "ScInterpreter::CalculateAddSub" ); 1159 ScMatrixRef pMat1 = NULL; 1160 ScMatrixRef pMat2 = NULL; 1161 double fVal1 = 0.0, fVal2 = 0.0; 1162 short nFmt1, nFmt2; 1163 nFmt1 = nFmt2 = NUMBERFORMAT_UNDEFINED; 1164 short nFmtCurrencyType = nCurFmtType; 1165 sal_uLong nFmtCurrencyIndex = nCurFmtIndex; 1166 short nFmtPercentType = nCurFmtType; 1167 if ( GetStackType() == svMatrix ) 1168 pMat2 = GetMatrix(); 1169 else 1170 { 1171 fVal2 = GetDouble(); 1172 switch ( nCurFmtType ) 1173 { 1174 case NUMBERFORMAT_DATE : 1175 case NUMBERFORMAT_TIME : 1176 case NUMBERFORMAT_DATETIME : 1177 nFmt2 = nCurFmtType; 1178 break; 1179 case NUMBERFORMAT_CURRENCY : 1180 nFmtCurrencyType = nCurFmtType; 1181 nFmtCurrencyIndex = nCurFmtIndex; 1182 break; 1183 case NUMBERFORMAT_PERCENT : 1184 nFmtPercentType = NUMBERFORMAT_PERCENT; 1185 break; 1186 } 1187 } 1188 if ( GetStackType() == svMatrix ) 1189 pMat1 = GetMatrix(); 1190 else 1191 { 1192 fVal1 = GetDouble(); 1193 switch ( nCurFmtType ) 1194 { 1195 case NUMBERFORMAT_DATE : 1196 case NUMBERFORMAT_TIME : 1197 case NUMBERFORMAT_DATETIME : 1198 nFmt1 = nCurFmtType; 1199 break; 1200 case NUMBERFORMAT_CURRENCY : 1201 nFmtCurrencyType = nCurFmtType; 1202 nFmtCurrencyIndex = nCurFmtIndex; 1203 break; 1204 case NUMBERFORMAT_PERCENT : 1205 nFmtPercentType = NUMBERFORMAT_PERCENT; 1206 break; 1207 } 1208 } 1209 if (pMat1 && pMat2) 1210 { 1211 ScMatrixRef pResMat; 1212 if ( _bSub ) 1213 { 1214 MatrixSub aSub; 1215 pResMat = lcl_MatrixCalculation(aSub ,pMat1, pMat2,this); 1216 } 1217 else 1218 { 1219 MatrixAdd aAdd; 1220 pResMat = lcl_MatrixCalculation(aAdd ,pMat1, pMat2,this); 1221 } 1222 1223 if (!pResMat) 1224 PushNoValue(); 1225 else 1226 PushMatrix(pResMat); 1227 } 1228 else if (pMat1 || pMat2) 1229 { 1230 double fVal; 1231 sal_Bool bFlag; 1232 ScMatrixRef pMat = pMat1; 1233 if (!pMat) 1234 { 1235 fVal = fVal1; 1236 pMat = pMat2; 1237 bFlag = sal_True; // double - Matrix 1238 } 1239 else 1240 { 1241 fVal = fVal2; 1242 bFlag = sal_False; // Matrix - double 1243 } 1244 SCSIZE nC, nR; 1245 pMat->GetDimensions(nC, nR); 1246 ScMatrixRef pResMat = GetNewMat(nC, nR); 1247 if (pResMat) 1248 { 1249 SCSIZE nCount = nC * nR; 1250 if (bFlag || !_bSub ) 1251 { 1252 for ( SCSIZE i = 0; i < nCount; i++ ) 1253 { 1254 if (pMat->IsValue(i)) 1255 pResMat->PutDouble( _bSub ? ::rtl::math::approxSub( fVal, pMat->GetDouble(i)) : ::rtl::math::approxAdd( pMat->GetDouble(i), fVal), i); 1256 else 1257 pResMat->PutString(ScGlobal::GetRscString(STR_NO_VALUE), i); 1258 } // for ( SCSIZE i = 0; i < nCount; i++ ) 1259 } // if (bFlag || !_bSub ) 1260 else 1261 { 1262 for ( SCSIZE i = 0; i < nCount; i++ ) 1263 { if (pMat->IsValue(i)) 1264 pResMat->PutDouble( ::rtl::math::approxSub( pMat->GetDouble(i), fVal), i); 1265 else 1266 pResMat->PutString(ScGlobal::GetRscString(STR_NO_VALUE), i); 1267 } // for ( SCSIZE i = 0; i < nCount; i++ ) 1268 } 1269 PushMatrix(pResMat); 1270 } 1271 else 1272 PushIllegalArgument(); 1273 } 1274 else if ( _bSub ) 1275 PushDouble( ::rtl::math::approxSub( fVal1, fVal2 ) ); 1276 else 1277 PushDouble( ::rtl::math::approxAdd( fVal1, fVal2 ) ); 1278 if ( nFmtCurrencyType == NUMBERFORMAT_CURRENCY ) 1279 { 1280 nFuncFmtType = nFmtCurrencyType; 1281 nFuncFmtIndex = nFmtCurrencyIndex; 1282 } 1283 else 1284 { 1285 lcl_GetDiffDateTimeFmtType( nFuncFmtType, nFmt1, nFmt2 ); 1286 if ( nFmtPercentType == NUMBERFORMAT_PERCENT && nFuncFmtType == NUMBERFORMAT_NUMBER ) 1287 nFuncFmtType = NUMBERFORMAT_PERCENT; 1288 } 1289 } 1290 1291 void ScInterpreter::ScAmpersand() 1292 { 1293 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "sc", "er", "ScInterpreter::ScAmpersand" ); 1294 ScMatrixRef pMat1 = NULL; 1295 ScMatrixRef pMat2 = NULL; 1296 String sStr1, sStr2; 1297 if ( GetStackType() == svMatrix ) 1298 pMat2 = GetMatrix(); 1299 else 1300 sStr2 = GetString(); 1301 if ( GetStackType() == svMatrix ) 1302 pMat1 = GetMatrix(); 1303 else 1304 sStr1 = GetString(); 1305 if (pMat1 && pMat2) 1306 { 1307 ScMatrixRef pResMat = MatConcat(pMat1, pMat2); 1308 if (!pResMat) 1309 PushNoValue(); 1310 else 1311 PushMatrix(pResMat); 1312 } 1313 else if (pMat1 || pMat2) 1314 { 1315 String sStr; 1316 sal_Bool bFlag; 1317 ScMatrixRef pMat = pMat1; 1318 if (!pMat) 1319 { 1320 sStr = sStr1; 1321 pMat = pMat2; 1322 bFlag = sal_True; // double - Matrix 1323 } 1324 else 1325 { 1326 sStr = sStr2; 1327 bFlag = sal_False; // Matrix - double 1328 } 1329 SCSIZE nC, nR; 1330 pMat->GetDimensions(nC, nR); 1331 ScMatrixRef pResMat = GetNewMat(nC, nR); 1332 if (pResMat) 1333 { 1334 SCSIZE nCount = nC * nR; 1335 if (nGlobalError) 1336 { 1337 for ( SCSIZE i = 0; i < nCount; i++ ) 1338 pResMat->PutError( nGlobalError, i); 1339 } 1340 else if (bFlag) 1341 { 1342 for ( SCSIZE i = 0; i < nCount; i++ ) 1343 { 1344 sal_uInt16 nErr = pMat->GetErrorIfNotString( i); 1345 if (nErr) 1346 pResMat->PutError( nErr, i); 1347 else 1348 { 1349 String aTmp( sStr); 1350 aTmp += pMat->GetString( *pFormatter, i); 1351 pResMat->PutString( aTmp, i); 1352 } 1353 } 1354 } 1355 else 1356 { 1357 for ( SCSIZE i = 0; i < nCount; i++ ) 1358 { 1359 sal_uInt16 nErr = pMat->GetErrorIfNotString( i); 1360 if (nErr) 1361 pResMat->PutError( nErr, i); 1362 else 1363 { 1364 String aTmp( pMat->GetString( *pFormatter, i)); 1365 aTmp += sStr; 1366 pResMat->PutString( aTmp, i); 1367 } 1368 } 1369 } 1370 PushMatrix(pResMat); 1371 } 1372 else 1373 PushIllegalArgument(); 1374 } 1375 else 1376 { 1377 if ( CheckStringResultLen( sStr1, sStr2 ) ) 1378 sStr1 += sStr2; 1379 PushString(sStr1); 1380 } 1381 } 1382 1383 void ScInterpreter::ScSub() 1384 { 1385 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "sc", "er", "ScInterpreter::ScSub" ); 1386 CalculateAddSub(sal_True); 1387 } 1388 1389 void ScInterpreter::ScMul() 1390 { 1391 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "sc", "er", "ScInterpreter::ScMul" ); 1392 ScMatrixRef pMat1 = NULL; 1393 ScMatrixRef pMat2 = NULL; 1394 double fVal1 = 0.0, fVal2 = 0.0; 1395 short nFmtCurrencyType = nCurFmtType; 1396 sal_uLong nFmtCurrencyIndex = nCurFmtIndex; 1397 if ( GetStackType() == svMatrix ) 1398 pMat2 = GetMatrix(); 1399 else 1400 { 1401 fVal2 = GetDouble(); 1402 switch ( nCurFmtType ) 1403 { 1404 case NUMBERFORMAT_CURRENCY : 1405 nFmtCurrencyType = nCurFmtType; 1406 nFmtCurrencyIndex = nCurFmtIndex; 1407 break; 1408 } 1409 } 1410 if ( GetStackType() == svMatrix ) 1411 pMat1 = GetMatrix(); 1412 else 1413 { 1414 fVal1 = GetDouble(); 1415 switch ( nCurFmtType ) 1416 { 1417 case NUMBERFORMAT_CURRENCY : 1418 nFmtCurrencyType = nCurFmtType; 1419 nFmtCurrencyIndex = nCurFmtIndex; 1420 break; 1421 } 1422 } 1423 if (pMat1 && pMat2) 1424 { 1425 MatrixMul aMul; 1426 ScMatrixRef pResMat = lcl_MatrixCalculation(aMul,pMat1, pMat2,this); 1427 if (!pResMat) 1428 PushNoValue(); 1429 else 1430 PushMatrix(pResMat); 1431 } 1432 else if (pMat1 || pMat2) 1433 { 1434 double fVal; 1435 ScMatrixRef pMat = pMat1; 1436 if (!pMat) 1437 { 1438 fVal = fVal1; 1439 pMat = pMat2; 1440 } 1441 else 1442 fVal = fVal2; 1443 SCSIZE nC, nR; 1444 pMat->GetDimensions(nC, nR); 1445 ScMatrixRef pResMat = GetNewMat(nC, nR); 1446 if (pResMat) 1447 { 1448 SCSIZE nCount = nC * nR; 1449 for ( SCSIZE i = 0; i < nCount; i++ ) 1450 if (pMat->IsValue(i)) 1451 pResMat->PutDouble(pMat->GetDouble(i)*fVal, i); 1452 else 1453 pResMat->PutString(ScGlobal::GetRscString(STR_NO_VALUE), i); 1454 PushMatrix(pResMat); 1455 } 1456 else 1457 PushIllegalArgument(); 1458 } 1459 else 1460 PushDouble(fVal1 * fVal2); 1461 if ( nFmtCurrencyType == NUMBERFORMAT_CURRENCY ) 1462 { 1463 nFuncFmtType = nFmtCurrencyType; 1464 nFuncFmtIndex = nFmtCurrencyIndex; 1465 } 1466 } 1467 1468 void ScInterpreter::ScDiv() 1469 { 1470 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "sc", "er", "ScInterpreter::ScDiv" ); 1471 ScMatrixRef pMat1 = NULL; 1472 ScMatrixRef pMat2 = NULL; 1473 double fVal1 = 0.0, fVal2 = 0.0; 1474 short nFmtCurrencyType = nCurFmtType; 1475 sal_uLong nFmtCurrencyIndex = nCurFmtIndex; 1476 short nFmtCurrencyType2 = NUMBERFORMAT_UNDEFINED; 1477 if ( GetStackType() == svMatrix ) 1478 pMat2 = GetMatrix(); 1479 else 1480 { 1481 fVal2 = GetDouble(); 1482 // hier kein Currency uebernehmen, 123kg/456DM sind nicht DM 1483 nFmtCurrencyType2 = nCurFmtType; 1484 } 1485 if ( GetStackType() == svMatrix ) 1486 pMat1 = GetMatrix(); 1487 else 1488 { 1489 fVal1 = GetDouble(); 1490 switch ( nCurFmtType ) 1491 { 1492 case NUMBERFORMAT_CURRENCY : 1493 nFmtCurrencyType = nCurFmtType; 1494 nFmtCurrencyIndex = nCurFmtIndex; 1495 break; 1496 } 1497 } 1498 if (pMat1 && pMat2) 1499 { 1500 MatrixDiv aDiv; 1501 ScMatrixRef pResMat = lcl_MatrixCalculation(aDiv,pMat1, pMat2,this); 1502 if (!pResMat) 1503 PushNoValue(); 1504 else 1505 PushMatrix(pResMat); 1506 } 1507 else if (pMat1 || pMat2) 1508 { 1509 double fVal; 1510 sal_Bool bFlag; 1511 ScMatrixRef pMat = pMat1; 1512 if (!pMat) 1513 { 1514 fVal = fVal1; 1515 pMat = pMat2; 1516 bFlag = sal_True; // double - Matrix 1517 } 1518 else 1519 { 1520 fVal = fVal2; 1521 bFlag = sal_False; // Matrix - double 1522 } 1523 SCSIZE nC, nR; 1524 pMat->GetDimensions(nC, nR); 1525 ScMatrixRef pResMat = GetNewMat(nC, nR); 1526 if (pResMat) 1527 { 1528 SCSIZE nCount = nC * nR; 1529 if (bFlag) 1530 { for ( SCSIZE i = 0; i < nCount; i++ ) 1531 if (pMat->IsValue(i)) 1532 pResMat->PutDouble( div( fVal, pMat->GetDouble(i)), i); 1533 else 1534 pResMat->PutString(ScGlobal::GetRscString(STR_NO_VALUE), i); 1535 } 1536 else 1537 { for ( SCSIZE i = 0; i < nCount; i++ ) 1538 if (pMat->IsValue(i)) 1539 pResMat->PutDouble( div( pMat->GetDouble(i), fVal), i); 1540 else 1541 pResMat->PutString(ScGlobal::GetRscString(STR_NO_VALUE), i); 1542 } 1543 PushMatrix(pResMat); 1544 } 1545 else 1546 PushIllegalArgument(); 1547 } 1548 else 1549 { 1550 PushDouble( div( fVal1, fVal2) ); 1551 } 1552 if ( nFmtCurrencyType == NUMBERFORMAT_CURRENCY && nFmtCurrencyType2 != NUMBERFORMAT_CURRENCY ) 1553 { // auch DM/DM ist nicht DM bzw. DEM/EUR nicht DEM 1554 nFuncFmtType = nFmtCurrencyType; 1555 nFuncFmtIndex = nFmtCurrencyIndex; 1556 } 1557 } 1558 1559 void ScInterpreter::ScPower() 1560 { 1561 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "sc", "er", "ScInterpreter::ScPower" ); 1562 if ( MustHaveParamCount( GetByte(), 2 ) ) 1563 ScPow(); 1564 } 1565 1566 void ScInterpreter::ScPow() 1567 { 1568 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "sc", "er", "ScInterpreter::ScPow" ); 1569 ScMatrixRef pMat1 = NULL; 1570 ScMatrixRef pMat2 = NULL; 1571 double fVal1 = 0.0, fVal2 = 0.0; 1572 if ( GetStackType() == svMatrix ) 1573 pMat2 = GetMatrix(); 1574 else 1575 fVal2 = GetDouble(); 1576 if ( GetStackType() == svMatrix ) 1577 pMat1 = GetMatrix(); 1578 else 1579 fVal1 = GetDouble(); 1580 if (pMat1 && pMat2) 1581 { 1582 MatrixPow aPow; 1583 ScMatrixRef pResMat = lcl_MatrixCalculation(aPow,pMat1, pMat2,this); 1584 if (!pResMat) 1585 PushNoValue(); 1586 else 1587 PushMatrix(pResMat); 1588 } 1589 else if (pMat1 || pMat2) 1590 { 1591 double fVal; 1592 sal_Bool bFlag; 1593 ScMatrixRef pMat = pMat1; 1594 if (!pMat) 1595 { 1596 fVal = fVal1; 1597 pMat = pMat2; 1598 bFlag = sal_True; // double - Matrix 1599 } 1600 else 1601 { 1602 fVal = fVal2; 1603 bFlag = sal_False; // Matrix - double 1604 } 1605 SCSIZE nC, nR; 1606 pMat->GetDimensions(nC, nR); 1607 ScMatrixRef pResMat = GetNewMat(nC, nR); 1608 if (pResMat) 1609 { 1610 SCSIZE nCount = nC * nR; 1611 if (bFlag) 1612 { for ( SCSIZE i = 0; i < nCount; i++ ) 1613 if (pMat->IsValue(i)) 1614 pResMat->PutDouble(pow(fVal,pMat->GetDouble(i)), i); 1615 else 1616 pResMat->PutString(ScGlobal::GetRscString(STR_NO_VALUE), i); 1617 } 1618 else 1619 { for ( SCSIZE i = 0; i < nCount; i++ ) 1620 if (pMat->IsValue(i)) 1621 pResMat->PutDouble(pow(pMat->GetDouble(i),fVal), i); 1622 else 1623 pResMat->PutString(ScGlobal::GetRscString(STR_NO_VALUE), i); 1624 } 1625 PushMatrix(pResMat); 1626 } 1627 else 1628 PushIllegalArgument(); 1629 } 1630 else 1631 PushDouble(pow(fVal1,fVal2)); 1632 } 1633 1634 void ScInterpreter::ScSumProduct() 1635 { 1636 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "sc", "er", "ScInterpreter::ScSumProduct" ); 1637 sal_uInt8 nParamCount = GetByte(); 1638 if ( !MustHaveParamCount( nParamCount, 1, 30 ) ) 1639 return; 1640 1641 ScMatrixRef pMat1 = NULL; 1642 ScMatrixRef pMat2 = NULL; 1643 ScMatrixRef pMat = NULL; 1644 pMat2 = GetMatrix(); 1645 if (!pMat2) 1646 { 1647 PushIllegalParameter(); 1648 return; 1649 } 1650 SCSIZE nC, nC1; 1651 SCSIZE nR, nR1; 1652 pMat2->GetDimensions(nC, nR); 1653 pMat = pMat2; 1654 MatrixMul aMul; 1655 for (sal_uInt16 i = 1; i < nParamCount; i++) 1656 { 1657 pMat1 = GetMatrix(); 1658 if (!pMat1) 1659 { 1660 PushIllegalParameter(); 1661 return; 1662 } 1663 pMat1->GetDimensions(nC1, nR1); 1664 if (nC1 != nC || nR1 != nR) 1665 { 1666 PushNoValue(); 1667 return; 1668 } 1669 ScMatrixRef pResMat = lcl_MatrixCalculation(aMul,pMat1, pMat,this); 1670 if (!pResMat) 1671 { 1672 PushNoValue(); 1673 return; 1674 } 1675 else 1676 pMat = pResMat; 1677 } 1678 double fSum = 0.0; 1679 SCSIZE nCount = pMat->GetElementCount(); 1680 for (SCSIZE j = 0; j < nCount; j++) 1681 { 1682 if (!pMat->IsString(j)) 1683 fSum += pMat->GetDouble(j); 1684 } 1685 PushDouble(fSum); 1686 } 1687 1688 void ScInterpreter::ScSumX2MY2() 1689 { 1690 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "sc", "er", "ScInterpreter::ScSumX2MY2" ); 1691 CalculateSumX2MY2SumX2DY2(sal_False); 1692 } 1693 void ScInterpreter::CalculateSumX2MY2SumX2DY2(sal_Bool _bSumX2DY2) 1694 { 1695 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "sc", "er", "ScInterpreter::CalculateSumX2MY2SumX2DY2" ); 1696 if ( !MustHaveParamCount( GetByte(), 2 ) ) 1697 return; 1698 1699 ScMatrixRef pMat1 = NULL; 1700 ScMatrixRef pMat2 = NULL; 1701 SCSIZE i, j; 1702 pMat2 = GetMatrix(); 1703 pMat1 = GetMatrix(); 1704 if (!pMat2 || !pMat1) 1705 { 1706 PushIllegalParameter(); 1707 return; 1708 } 1709 SCSIZE nC1, nC2; 1710 SCSIZE nR1, nR2; 1711 pMat2->GetDimensions(nC2, nR2); 1712 pMat1->GetDimensions(nC1, nR1); 1713 if (nC1 != nC2 || nR1 != nR2) 1714 { 1715 PushNoValue(); 1716 return; 1717 } 1718 double fVal, fSum = 0.0; 1719 for (i = 0; i < nC1; i++) 1720 for (j = 0; j < nR1; j++) 1721 if (!pMat1->IsString(i,j) && !pMat2->IsString(i,j)) 1722 { 1723 fVal = pMat1->GetDouble(i,j); 1724 fSum += fVal * fVal; 1725 fVal = pMat2->GetDouble(i,j); 1726 if ( _bSumX2DY2 ) 1727 fSum += fVal * fVal; 1728 else 1729 fSum -= fVal * fVal; 1730 } 1731 PushDouble(fSum); 1732 } 1733 1734 void ScInterpreter::ScSumX2DY2() 1735 { 1736 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "sc", "er", "ScInterpreter::ScSumX2DY2" ); 1737 CalculateSumX2MY2SumX2DY2(sal_True); 1738 } 1739 1740 void ScInterpreter::ScSumXMY2() 1741 { 1742 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "sc", "er", "ScInterpreter::ScSumXMY2" ); 1743 if ( !MustHaveParamCount( GetByte(), 2 ) ) 1744 return; 1745 1746 ScMatrixRef pMat1 = NULL; 1747 ScMatrixRef pMat2 = NULL; 1748 pMat2 = GetMatrix(); 1749 pMat1 = GetMatrix(); 1750 if (!pMat2 || !pMat1) 1751 { 1752 PushIllegalParameter(); 1753 return; 1754 } 1755 SCSIZE nC1, nC2; 1756 SCSIZE nR1, nR2; 1757 pMat2->GetDimensions(nC2, nR2); 1758 pMat1->GetDimensions(nC1, nR1); 1759 if (nC1 != nC2 || nR1 != nR2) 1760 { 1761 PushNoValue(); 1762 return; 1763 } // if (nC1 != nC2 || nR1 != nR2) 1764 MatrixSub aSub; 1765 ScMatrixRef pResMat = lcl_MatrixCalculation(aSub,pMat1, pMat2,this); 1766 if (!pResMat) 1767 { 1768 PushNoValue(); 1769 } 1770 else 1771 { 1772 double fVal, fSum = 0.0; 1773 SCSIZE nCount = pResMat->GetElementCount(); 1774 for (SCSIZE i = 0; i < nCount; i++) 1775 if (!pResMat->IsString(i)) 1776 { 1777 fVal = pResMat->GetDouble(i); 1778 fSum += fVal * fVal; 1779 } 1780 PushDouble(fSum); 1781 } 1782 } 1783 1784 void ScInterpreter::ScFrequency() 1785 { 1786 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "sc", "er", "ScInterpreter::ScFrequency" ); 1787 if ( !MustHaveParamCount( GetByte(), 2 ) ) 1788 return; 1789 1790 vector<double> aBinArray; 1791 vector<long> aBinIndexOrder; 1792 1793 GetSortArray(1, aBinArray, &aBinIndexOrder); 1794 SCSIZE nBinSize = aBinArray.size(); 1795 if (nGlobalError) 1796 { 1797 PushNoValue(); 1798 return; 1799 } 1800 1801 vector<double> aDataArray; 1802 GetSortArray(1, aDataArray); 1803 SCSIZE nDataSize = aDataArray.size(); 1804 1805 if (aDataArray.empty() || nGlobalError) 1806 { 1807 PushNoValue(); 1808 return; 1809 } 1810 ScMatrixRef pResMat = GetNewMat(1, nBinSize+1); 1811 if (!pResMat) 1812 { 1813 PushIllegalArgument(); 1814 return; 1815 } 1816 1817 if (nBinSize != aBinIndexOrder.size()) 1818 { 1819 PushIllegalArgument(); 1820 return; 1821 } 1822 1823 SCSIZE j; 1824 SCSIZE i = 0; 1825 for (j = 0; j < nBinSize; ++j) 1826 { 1827 SCSIZE nCount = 0; 1828 while (i < nDataSize && aDataArray[i] <= aBinArray[j]) 1829 { 1830 ++nCount; 1831 ++i; 1832 } 1833 pResMat->PutDouble(static_cast<double>(nCount), aBinIndexOrder[j]); 1834 } 1835 pResMat->PutDouble(static_cast<double>(nDataSize-i), j); 1836 PushMatrix(pResMat); 1837 } 1838 1839 // ----------------------------------------------------------------------------- 1840 // Helper methods for LINEST/LOGEST and TREND/GROWTH 1841 // All matrices must already exist and have the needed size, no control tests 1842 // done. Those methodes, which names start with lcl_T, are adapted to case 3, 1843 // where Y (=observed values) is given as row. 1844 // Remember, ScMatrix matrices are zero based, index access (column,row). 1845 // ----------------------------------------------------------------------------- 1846 1847 // Multiply n x m Mat A with m x l Mat B to n x l Mat R 1848 void lcl_MFastMult( ScMatrixRef pA, ScMatrixRef pB, ScMatrixRef pR, SCSIZE n, SCSIZE m, SCSIZE l ) 1849 { 1850 double sum; 1851 for (SCSIZE row = 0; row < n; row++) 1852 { 1853 for (SCSIZE col = 0; col < l; col++) 1854 { // result element(col, row) =sum[ (row of A) * (column of B)] 1855 sum = 0.0; 1856 for (SCSIZE k = 0; k < m; k++) 1857 sum += pA->GetDouble(k,row) * pB->GetDouble(col,k); 1858 pR->PutDouble(sum, col, row); 1859 } 1860 } 1861 } 1862 1863 // <A;B> over all elements; uses the matrices as vectors of length M 1864 double lcl_GetSumProduct( ScMatrixRef pMatA, ScMatrixRef pMatB, SCSIZE nM ) 1865 { 1866 double fSum = 0.0; 1867 for (SCSIZE i=0; i<nM; i++) 1868 fSum += pMatA->GetDouble(i) * pMatB->GetDouble(i); 1869 return fSum; 1870 } 1871 1872 // Special version for use within QR decomposition. 1873 // Euclidean norm of column index C starting in row index R; 1874 // matrix A has count N rows. 1875 double lcl_GetColumnEuclideanNorm( ScMatrixRef pMatA, SCSIZE nC, SCSIZE nR, SCSIZE nN ) 1876 { 1877 double fNorm = 0.0; 1878 for (SCSIZE row=nR; row<nN; row++) 1879 fNorm += (pMatA->GetDouble(nC,row)) * (pMatA->GetDouble(nC,row)); 1880 return sqrt(fNorm); 1881 } 1882 1883 // Euclidean norm of row index R starting in column index C; 1884 // matrix A has count N columns. 1885 double lcl_TGetColumnEuclideanNorm( ScMatrixRef pMatA, SCSIZE nR, SCSIZE nC, SCSIZE nN ) 1886 { 1887 double fNorm = 0.0; 1888 for (SCSIZE col=nC; col<nN; col++) 1889 fNorm += (pMatA->GetDouble(col,nR)) * (pMatA->GetDouble(col,nR)); 1890 return sqrt(fNorm); 1891 } 1892 1893 // Special version for use within QR decomposition. 1894 // Maximum norm of column index C starting in row index R; 1895 // matrix A has count N rows. 1896 double lcl_GetColumnMaximumNorm( ScMatrixRef pMatA, SCSIZE nC, SCSIZE nR, SCSIZE nN ) 1897 { 1898 double fNorm = 0.0; 1899 for (SCSIZE row=nR; row<nN; row++) 1900 if (fNorm < fabs(pMatA->GetDouble(nC,row))) 1901 fNorm = fabs(pMatA->GetDouble(nC,row)); 1902 return fNorm; 1903 } 1904 1905 // Maximum norm of row index R starting in col index C; 1906 // matrix A has count N columns. 1907 double lcl_TGetColumnMaximumNorm( ScMatrixRef pMatA, SCSIZE nR, SCSIZE nC, SCSIZE nN ) 1908 { 1909 double fNorm = 0.0; 1910 for (SCSIZE col=nC; col<nN; col++) 1911 if (fNorm < fabs(pMatA->GetDouble(col,nR))) 1912 fNorm = fabs(pMatA->GetDouble(col,nR)); 1913 return fNorm; 1914 } 1915 1916 // Special version for use within QR decomposition. 1917 // <A(Ca);B(Cb)> starting in row index R; 1918 // Ca and Cb are indices of columns, matrices A and B have count N rows. 1919 double lcl_GetColumnSumProduct( ScMatrixRef pMatA, SCSIZE nCa, ScMatrixRef pMatB, 1920 SCSIZE nCb, SCSIZE nR, SCSIZE nN ) 1921 { 1922 double fResult = 0.0; 1923 for (SCSIZE row=nR; row<nN; row++) 1924 fResult += pMatA->GetDouble(nCa,row) * pMatB->GetDouble(nCb,row); 1925 return fResult; 1926 } 1927 1928 // <A(Ra);B(Rb)> starting in column index C; 1929 // Ra and Rb are indices of rows, matrices A and B have count N columns. 1930 double lcl_TGetColumnSumProduct( ScMatrixRef pMatA, SCSIZE nRa, 1931 ScMatrixRef pMatB, SCSIZE nRb, SCSIZE nC, SCSIZE nN ) 1932 { 1933 double fResult = 0.0; 1934 for (SCSIZE col=nC; col<nN; col++) 1935 fResult += pMatA->GetDouble(col,nRa) * pMatB->GetDouble(col,nRb); 1936 return fResult; 1937 } 1938 1939 // #118029# no mathematical signum, but used to switch between adding and subtracting 1940 double lcl_GetSign(double fValue) 1941 { 1942 return (fValue >= 0.0 ? 1.0 : -1.0 ); 1943 } 1944 1945 /* Calculates a QR decomposition with Householder reflection. 1946 * For each NxK matrix A exists a decomposition A=Q*R with an orthogonal 1947 * NxN matrix Q and a NxK matrix R. 1948 * Q=H1*H2*...*Hk with Householder matrices H. Such a householder matrix can 1949 * be build from a vector u by H=I-(2/u'u)*(u u'). This vectors u are returned 1950 * in the columns of matrix A, overwriting the old content. 1951 * The matrix R has a quadric upper part KxK with values in the upper right 1952 * triangle and zeros in all other elements. Here the diagonal elements of R 1953 * are stored in the vector R and the other upper right elements in the upper 1954 * right of the matrix A. 1955 * The function returns false, if calculation breaks. But because of round-off 1956 * errors singularity is often not detected. 1957 */ 1958 bool lcl_CalculateQRdecomposition(ScMatrixRef pMatA, 1959 ::std::vector< double>& pVecR, SCSIZE nK, SCSIZE nN) 1960 { 1961 double fScale ; 1962 double fEuclid ; 1963 double fFactor ; 1964 double fSignum ; 1965 double fSum ; 1966 // ScMatrix matrices are zero based, index access (column,row) 1967 for (SCSIZE col = 0; col <nK; col++) 1968 { 1969 // calculate vector u of the householder transformation 1970 fScale = lcl_GetColumnMaximumNorm(pMatA, col, col, nN); 1971 if (fScale == 0.0) 1972 // A is singular 1973 return false; 1974 1975 for (SCSIZE row = col; row <nN; row++) 1976 pMatA->PutDouble( pMatA->GetDouble(col,row)/fScale, col, row); 1977 1978 fEuclid = lcl_GetColumnEuclideanNorm(pMatA, col, col, nN); 1979 fFactor = 1.0/fEuclid/(fEuclid + fabs(pMatA->GetDouble(col,col))); 1980 fSignum = lcl_GetSign(pMatA->GetDouble(col,col)); 1981 pMatA->PutDouble( pMatA->GetDouble(col,col) + fSignum*fEuclid, col,col); 1982 pVecR[col] = -fSignum * fScale * fEuclid; 1983 1984 // apply Householder transformation to A 1985 for (SCSIZE c=col+1; c<nK; c++) 1986 { 1987 fSum =lcl_GetColumnSumProduct(pMatA, col, pMatA, c, col, nN); 1988 for (SCSIZE row = col; row <nN; row++) 1989 pMatA->PutDouble( pMatA->GetDouble(c,row) 1990 - fSum * fFactor * pMatA->GetDouble(col,row), c, row); 1991 } 1992 } 1993 return true; 1994 } 1995 1996 // same with transposed matrix A, N is count of columns, K count of rows 1997 bool lcl_TCalculateQRdecomposition(ScMatrixRef pMatA, 1998 ::std::vector< double>& pVecR, SCSIZE nK, SCSIZE nN) 1999 { 2000 double fScale ; 2001 double fEuclid ; 2002 double fFactor ; 2003 double fSignum ; 2004 double fSum ; 2005 // ScMatrix matrices are zero based, index access (column,row) 2006 for (SCSIZE row = 0; row <nK; row++) 2007 { 2008 // calculate vector u of the householder transformation 2009 fScale = lcl_TGetColumnMaximumNorm(pMatA, row, row, nN); 2010 if (fScale == 0.0) 2011 // A is singular 2012 return false; 2013 2014 for (SCSIZE col = row; col <nN; col++) 2015 pMatA->PutDouble( pMatA->GetDouble(col,row)/fScale, col, row); 2016 2017 fEuclid = lcl_TGetColumnEuclideanNorm(pMatA, row, row, nN); 2018 fFactor = 1.0/fEuclid/(fEuclid + fabs(pMatA->GetDouble(row,row))); 2019 fSignum = lcl_GetSign(pMatA->GetDouble(row,row)); 2020 pMatA->PutDouble( pMatA->GetDouble(row,row) + fSignum*fEuclid, row,row); 2021 pVecR[row] = -fSignum * fScale * fEuclid; 2022 2023 // apply Householder transformation to A 2024 for (SCSIZE r=row+1; r<nK; r++) 2025 { 2026 fSum =lcl_TGetColumnSumProduct(pMatA, row, pMatA, r, row, nN); 2027 for (SCSIZE col = row; col <nN; col++) 2028 pMatA->PutDouble( pMatA->GetDouble(col,r) 2029 - fSum * fFactor * pMatA->GetDouble(col,row), col, r); 2030 } 2031 } 2032 return true; 2033 } 2034 2035 2036 /* Applies a Householder transformation to a column vector Y with is given as 2037 * Nx1 Matrix. The Vektor u, from which the Householder transformation is build, 2038 * is the column part in matrix A, with column index C, starting with row 2039 * index C. A is the result of the QR decomposition as obtained from 2040 * lcl_CaluclateQRdecomposition. 2041 */ 2042 void lcl_ApplyHouseholderTransformation(ScMatrixRef pMatA, SCSIZE nC, 2043 ScMatrixRef pMatY, SCSIZE nN) 2044 { 2045 // ScMatrix matrices are zero based, index access (column,row) 2046 double fDenominator = lcl_GetColumnSumProduct(pMatA, nC, pMatA, nC, nC, nN); 2047 double fNumerator = lcl_GetColumnSumProduct(pMatA, nC, pMatY, 0, nC, nN); 2048 double fFactor = 2.0 * (fNumerator/fDenominator); 2049 for (SCSIZE row = nC; row < nN; row++) 2050 pMatY->PutDouble( 2051 pMatY->GetDouble(row) - fFactor * pMatA->GetDouble(nC,row), row); 2052 } 2053 2054 // Same with transposed matrices A and Y. 2055 void lcl_TApplyHouseholderTransformation(ScMatrixRef pMatA, SCSIZE nR, 2056 ScMatrixRef pMatY, SCSIZE nN) 2057 { 2058 // ScMatrix matrices are zero based, index access (column,row) 2059 double fDenominator = lcl_TGetColumnSumProduct(pMatA, nR, pMatA, nR, nR, nN); 2060 double fNumerator = lcl_TGetColumnSumProduct(pMatA, nR, pMatY, 0, nR, nN); 2061 double fFactor = 2.0 * (fNumerator/fDenominator); 2062 for (SCSIZE col = nR; col < nN; col++) 2063 pMatY->PutDouble( 2064 pMatY->GetDouble(col) - fFactor * pMatA->GetDouble(col,nR), col); 2065 } 2066 2067 /* Solve for X in R*X=S using back substitution. The solution X overwrites S. 2068 * Uses R from the result of the QR decomposition of a NxK matrix A. 2069 * S is a column vector given as matrix, with at least elements on index 2070 * 0 to K-1; elements on index>=K are ignored. Vector R must not have zero 2071 * elements, no check is done. 2072 */ 2073 void lcl_SolveWithUpperRightTriangle(ScMatrixRef pMatA, 2074 ::std::vector< double>& pVecR, ScMatrixRef pMatS, 2075 SCSIZE nK, bool bIsTransposed) 2076 { 2077 // ScMatrix matrices are zero based, index access (column,row) 2078 double fSum; 2079 SCSIZE row; 2080 // SCSIZE is never negative, therefore test with rowp1=row+1 2081 for (SCSIZE rowp1 = nK; rowp1>0; rowp1--) 2082 { 2083 row = rowp1-1; 2084 fSum = pMatS->GetDouble(row); 2085 for (SCSIZE col = rowp1; col<nK ; col++) 2086 if (bIsTransposed) 2087 fSum -= pMatA->GetDouble(row,col) * pMatS->GetDouble(col); 2088 else 2089 fSum -= pMatA->GetDouble(col,row) * pMatS->GetDouble(col); 2090 pMatS->PutDouble( fSum / pVecR[row] , row); 2091 } 2092 } 2093 2094 /* Solve for X in R' * X= T using forward substitution. The solution X 2095 * overwrites T. Uses R from the result of the QR decomposition of a NxK 2096 * matrix A. T is a column vectors given as matrix, with at least elements on 2097 * index 0 to K-1; elements on index>=K are ignored. Vector R must not have 2098 * zero elements, no check is done. 2099 */ 2100 void lcl_SolveWithLowerLeftTriangle(ScMatrixRef pMatA, 2101 ::std::vector< double>& pVecR, ScMatrixRef pMatT, 2102 SCSIZE nK, bool bIsTransposed) 2103 { 2104 // ScMatrix matrices are zero based, index access (column,row) 2105 double fSum; 2106 for (SCSIZE row = 0; row < nK; row++) 2107 { 2108 fSum = pMatT -> GetDouble(row); 2109 for (SCSIZE col=0; col < row; col++) 2110 { 2111 if (bIsTransposed) 2112 fSum -= pMatA->GetDouble(col,row) * pMatT->GetDouble(col); 2113 else 2114 fSum -= pMatA->GetDouble(row,col) * pMatT->GetDouble(col); 2115 } 2116 pMatT->PutDouble( fSum / pVecR[row] , row); 2117 } 2118 } 2119 2120 /* Calculates Z = R * B 2121 * R is given in matrix A and vector VecR as obtained from the QR 2122 * decompostion in lcl_CalculateQRdecomposition. B and Z are column vectors 2123 * given as matrix with at least index 0 to K-1; elements on index>=K are 2124 * not used. 2125 */ 2126 void lcl_ApplyUpperRightTriangle(ScMatrixRef pMatA, 2127 ::std::vector< double>& pVecR, ScMatrixRef pMatB, 2128 ScMatrixRef pMatZ, SCSIZE nK, bool bIsTransposed) 2129 { 2130 // ScMatrix matrices are zero based, index access (column,row) 2131 double fSum; 2132 for (SCSIZE row = 0; row < nK; row++) 2133 { 2134 fSum = pVecR[row] * pMatB->GetDouble(row); 2135 for (SCSIZE col = row+1; col < nK; col++) 2136 if (bIsTransposed) 2137 fSum += pMatA->GetDouble(row,col) * pMatB->GetDouble(col); 2138 else 2139 fSum += pMatA->GetDouble(col,row) * pMatB->GetDouble(col); 2140 pMatZ->PutDouble( fSum, row); 2141 } 2142 } 2143 2144 2145 2146 double lcl_GetMeanOverAll(ScMatrixRef pMat, SCSIZE nN) 2147 { 2148 double fSum = 0.0; 2149 for (SCSIZE i=0 ; i<nN; i++) 2150 fSum += pMat->GetDouble(i); 2151 return fSum/static_cast<double>(nN); 2152 } 2153 2154 // Calculates means of the columns of matrix X. X is a RxC matrix; 2155 // ResMat is a 1xC matrix (=row). 2156 void lcl_CalculateColumnMeans(ScMatrixRef pX, ScMatrixRef pResMat, 2157 SCSIZE nC, SCSIZE nR) 2158 { 2159 double fSum = 0.0; 2160 for (SCSIZE i=0; i < nC; i++) 2161 { 2162 fSum =0.0; 2163 for (SCSIZE k=0; k < nR; k++) 2164 fSum += pX->GetDouble(i,k); // GetDouble(Column,Row) 2165 pResMat ->PutDouble( fSum/static_cast<double>(nR),i); 2166 } 2167 } 2168 2169 // Calculates means of the rows of matrix X. X is a RxC matrix; 2170 // ResMat is a Rx1 matrix (=column). 2171 void lcl_CalculateRowMeans(ScMatrixRef pX, ScMatrixRef pResMat, 2172 SCSIZE nC, SCSIZE nR) 2173 { 2174 double fSum = 0.0; 2175 for (SCSIZE k=0; k < nR; k++) 2176 { 2177 fSum =0.0; 2178 for (SCSIZE i=0; i < nC; i++) 2179 fSum += pX->GetDouble(i,k); // GetDouble(Column,Row) 2180 pResMat ->PutDouble( fSum/static_cast<double>(nC),k); 2181 } 2182 } 2183 2184 void lcl_CalculateColumnsDelta(ScMatrixRef pMat, ScMatrixRef pColumnMeans, 2185 SCSIZE nC, SCSIZE nR) 2186 { 2187 for (SCSIZE i = 0; i < nC; i++) 2188 for (SCSIZE k = 0; k < nR; k++) 2189 pMat->PutDouble( ::rtl::math::approxSub 2190 (pMat->GetDouble(i,k) , pColumnMeans->GetDouble(i) ) , i, k); 2191 } 2192 2193 void lcl_CalculateRowsDelta(ScMatrixRef pMat, ScMatrixRef pRowMeans, 2194 SCSIZE nC, SCSIZE nR) 2195 { 2196 for (SCSIZE k = 0; k < nR; k++) 2197 for (SCSIZE i = 0; i < nC; i++) 2198 pMat->PutDouble( ::rtl::math::approxSub 2199 ( pMat->GetDouble(i,k) , pRowMeans->GetDouble(k) ) , i, k); 2200 } 2201 2202 // Case1 = simple regression 2203 // MatX = X - MeanX, MatY = Y - MeanY, y - haty = (y - MeanY) - (haty - MeanY) 2204 // = (y-MeanY)-((slope*x+a)-(slope*MeanX+a)) = (y-MeanY)-slope*(x-MeanX) 2205 double lcl_GetSSresid(ScMatrixRef pMatX, ScMatrixRef pMatY, double fSlope, 2206 SCSIZE nN) 2207 { 2208 double fSum = 0.0; 2209 double fTemp = 0.0; 2210 for (SCSIZE i=0; i<nN; i++) 2211 { 2212 fTemp = pMatY->GetDouble(i) - fSlope * pMatX->GetDouble(i); 2213 fSum += fTemp * fTemp; 2214 } 2215 return fSum; 2216 } 2217 2218 // Fill default values in matrix X, transform Y to log(Y) in case LOGEST|GROWTH, 2219 // determine sizes of matrices X and Y, determine kind of regression, clone 2220 // Y in case LOGEST|GROWTH, if constant. 2221 bool ScInterpreter::CheckMatrix(bool _bLOG, sal_uInt8& nCase, SCSIZE& nCX, 2222 SCSIZE& nCY, SCSIZE& nRX, SCSIZE& nRY, SCSIZE& M, 2223 SCSIZE& N, ScMatrixRef& pMatX, ScMatrixRef& pMatY) 2224 { 2225 2226 nCX = 0; 2227 nCY = 0; 2228 nRX = 0; 2229 nRY = 0; 2230 M = 0; 2231 N = 0; 2232 pMatY->GetDimensions(nCY, nRY); 2233 const SCSIZE nCountY = nCY * nRY; 2234 for ( SCSIZE i = 0; i < nCountY; i++ ) 2235 { 2236 if (!pMatY->IsValue(i)) 2237 { 2238 PushIllegalArgument(); 2239 return false; 2240 } 2241 } 2242 2243 if ( _bLOG ) 2244 { 2245 ScMatrixRef pNewY = pMatY->CloneIfConst(); 2246 for (SCSIZE nElem = 0; nElem < nCountY; nElem++) 2247 { 2248 const double fVal = pNewY->GetDouble(nElem); 2249 if (fVal <= 0.0) 2250 { 2251 PushIllegalArgument(); 2252 return false; 2253 } 2254 else 2255 pNewY->PutDouble(log(fVal), nElem); 2256 } 2257 pMatY = pNewY; 2258 } 2259 2260 if (pMatX) 2261 { 2262 pMatX->GetDimensions(nCX, nRX); 2263 const SCSIZE nCountX = nCX * nRX; 2264 for ( SCSIZE i = 0; i < nCountX; i++ ) 2265 if (!pMatX->IsValue(i)) 2266 { 2267 PushIllegalArgument(); 2268 return false; 2269 } 2270 if (nCX == nCY && nRX == nRY) 2271 { 2272 nCase = 1; // simple regression 2273 M = 1; 2274 N = nCountY; 2275 } 2276 else if (nCY != 1 && nRY != 1) 2277 { 2278 PushIllegalArgument(); 2279 return false; 2280 } 2281 else if (nCY == 1) 2282 { 2283 if (nRX != nRY) 2284 { 2285 PushIllegalArgument(); 2286 return false; 2287 } 2288 else 2289 { 2290 nCase = 2; // Y is column 2291 N = nRY; 2292 M = nCX; 2293 } 2294 } 2295 else if (nCX != nCY) 2296 { 2297 PushIllegalArgument(); 2298 return false; 2299 } 2300 else 2301 { 2302 nCase = 3; // Y is row 2303 N = nCY; 2304 M = nRX; 2305 } 2306 } 2307 else 2308 { 2309 pMatX = GetNewMat(nCY, nRY); 2310 nCX = nCY; 2311 nRX = nRY; 2312 if (!pMatX) 2313 { 2314 PushIllegalArgument(); 2315 return false; 2316 } 2317 for ( SCSIZE i = 1; i <= nCountY; i++ ) 2318 pMatX->PutDouble(static_cast<double>(i), i-1); 2319 nCase = 1; 2320 N = nCountY; 2321 M = 1; 2322 } 2323 return true; 2324 } 2325 // ----------------------------------------------------------------------------- 2326 2327 // LINEST 2328 void ScInterpreter::ScRGP() 2329 { 2330 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "sc", "er", "ScInterpreter::ScRGP" ); 2331 CalulateRGPRKP(false); 2332 } 2333 2334 // LOGEST 2335 void ScInterpreter::ScRKP() 2336 { 2337 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "sc", "er", "ScInterpreter::ScRKP" ); 2338 CalulateRGPRKP(true); 2339 } 2340 2341 void ScInterpreter::CalulateRGPRKP(bool _bRKP) 2342 { 2343 sal_uInt8 nParamCount = GetByte(); 2344 if ( !MustHaveParamCount( nParamCount, 1, 4 ) ) 2345 return; 2346 bool bConstant, bStats; 2347 2348 // optional forth parameter 2349 if (nParamCount == 4) 2350 bStats = GetBool(); 2351 else 2352 bStats = false; 2353 2354 // The third parameter may not be missing in ODF, if the forth parameter 2355 // is present. But Excel allows it with default true, we too. 2356 if (nParamCount >= 3) 2357 { 2358 if (IsMissing()) 2359 { 2360 Pop(); 2361 bConstant = true; 2362 // PushIllegalParameter(); if ODF behavior is desired 2363 // return; 2364 } 2365 else 2366 bConstant = GetBool(); 2367 } 2368 else 2369 bConstant = true; 2370 2371 ScMatrixRef pMatX; 2372 if (nParamCount >= 2) 2373 { 2374 if (IsMissing()) 2375 { 2376 // In ODF1.2 empty second parameter (which is two ;; ) is allowed 2377 Pop(); 2378 pMatX = NULL; 2379 } 2380 else 2381 { 2382 pMatX = GetMatrix(); 2383 } 2384 } 2385 else 2386 pMatX = NULL; 2387 2388 ScMatrixRef pMatY; 2389 pMatY = GetMatrix(); 2390 if (!pMatY) 2391 { 2392 PushIllegalParameter(); 2393 return; 2394 } 2395 2396 // 1 = simple; 2 = multiple with Y as column; 3 = multiple with Y as row 2397 sal_uInt8 nCase; 2398 2399 SCSIZE nCX, nCY; // number of columns 2400 SCSIZE nRX, nRY; // number of rows 2401 SCSIZE K = 0, N = 0; // K=number of variables X, N=number of data samples 2402 if ( !CheckMatrix(_bRKP,nCase,nCX,nCY,nRX,nRY,K,N,pMatX,pMatY) ) 2403 { 2404 PushIllegalParameter(); 2405 return; 2406 } 2407 2408 // Enough data samples? 2409 if ( (bConstant && (N<K+1)) || (!bConstant && (N<K)) || (N<1) || (K<1) ) 2410 { 2411 PushIllegalParameter(); 2412 return; 2413 } 2414 2415 ScMatrixRef pResMat; 2416 if (bStats) 2417 pResMat = GetNewMat(K+1,5); 2418 else 2419 pResMat = GetNewMat(K+1,1); 2420 if (!pResMat) 2421 { 2422 PushError(errCodeOverflow); 2423 return; 2424 } 2425 // Fill unused cells in pResMat; order (column,row) 2426 if (bStats) 2427 { 2428 for (SCSIZE i=2; i<K+1; i++) 2429 { 2430 pResMat->PutString(ScGlobal::GetRscString(STR_NV_STR), i, 2 ); 2431 pResMat->PutString(ScGlobal::GetRscString(STR_NV_STR), i, 3 ); 2432 pResMat->PutString(ScGlobal::GetRscString(STR_NV_STR), i, 4 ); 2433 } 2434 } 2435 2436 // Uses sum(x-MeanX)^2 and not [sum x^2]-N * MeanX^2 in case bConstant. 2437 // Clone constant matrices, so that Mat = Mat - Mean is possible. 2438 double fMeanY = 0.0; 2439 if (bConstant) 2440 { 2441 ScMatrixRef pNewX = pMatX->CloneIfConst(); 2442 ScMatrixRef pNewY = pMatY->CloneIfConst(); 2443 if (!pNewX || !pNewY) 2444 { 2445 PushError(errCodeOverflow); 2446 return; 2447 } 2448 pMatX = pNewX; 2449 pMatY = pNewY; 2450 // DeltaY is possible here; DeltaX depends on nCase, so later 2451 fMeanY = lcl_GetMeanOverAll(pMatY, N); 2452 for (SCSIZE i=0; i<N; i++) 2453 { 2454 pMatY->PutDouble( ::rtl::math::approxSub(pMatY->GetDouble(i),fMeanY), i ); 2455 } 2456 } 2457 2458 if (nCase==1) 2459 { 2460 // calculate simple regression 2461 double fMeanX = 0.0; 2462 if (bConstant) 2463 { // Mat = Mat - Mean 2464 fMeanX = lcl_GetMeanOverAll(pMatX, N); 2465 for (SCSIZE i=0; i<N; i++) 2466 { 2467 pMatX->PutDouble( ::rtl::math::approxSub(pMatX->GetDouble(i),fMeanX), i ); 2468 } 2469 } 2470 double fSumXY = lcl_GetSumProduct(pMatX,pMatY,N); 2471 double fSumX2 = lcl_GetSumProduct(pMatX,pMatX,N); 2472 if (fSumX2==0.0) 2473 { 2474 PushNoValue(); // all x-values are identical 2475 return; 2476 } 2477 double fSlope = fSumXY / fSumX2; 2478 double fIntercept = 0.0; 2479 if (bConstant) 2480 fIntercept = fMeanY - fSlope * fMeanX; 2481 pResMat->PutDouble(_bRKP ? exp(fIntercept) : fIntercept, 1, 0); //order (column,row) 2482 pResMat->PutDouble(_bRKP ? exp(fSlope) : fSlope, 0, 0); 2483 2484 if (bStats) 2485 { 2486 double fSSreg = fSlope * fSlope * fSumX2; 2487 pResMat->PutDouble(fSSreg, 0, 4); 2488 2489 double fDegreesFreedom =static_cast<double>( (bConstant) ? N-2 : N-1 ); 2490 pResMat->PutDouble(fDegreesFreedom, 1, 3); 2491 2492 double fSSresid = lcl_GetSSresid(pMatX,pMatY,fSlope,N); 2493 pResMat->PutDouble(fSSresid, 1, 4); 2494 2495 if (fDegreesFreedom == 0.0 || fSSresid == 0.0 || fSSreg == 0.0) 2496 { // exact fit; test SSreg too, because SSresid might be 2497 // unequal zero due to round of errors 2498 pResMat->PutDouble(0.0, 1, 4); // SSresid 2499 pResMat->PutString(ScGlobal::GetRscString(STR_NV_STR), 0, 3); // F 2500 pResMat->PutDouble(0.0, 1, 2); // RMSE 2501 pResMat->PutDouble(0.0, 0, 1); // SigmaSlope 2502 if (bConstant) 2503 pResMat->PutDouble(0.0, 1, 1); //SigmaIntercept 2504 else 2505 pResMat->PutString(ScGlobal::GetRscString(STR_NV_STR), 1, 1); 2506 pResMat->PutDouble(1.0, 0, 2); // R^2 2507 } 2508 else 2509 { 2510 double fFstatistic = (fSSreg / static_cast<double>(K)) 2511 / (fSSresid / fDegreesFreedom); 2512 pResMat->PutDouble(fFstatistic, 0, 3); 2513 2514 // standard error of estimate 2515 double fRMSE = sqrt(fSSresid / fDegreesFreedom); 2516 pResMat->PutDouble(fRMSE, 1, 2); 2517 2518 double fSigmaSlope = fRMSE / sqrt(fSumX2); 2519 pResMat->PutDouble(fSigmaSlope, 0, 1); 2520 2521 if (bConstant) 2522 { 2523 double fSigmaIntercept = fRMSE 2524 * sqrt(fMeanX*fMeanX/fSumX2 + 1.0/static_cast<double>(N)); 2525 pResMat->PutDouble(fSigmaIntercept, 1, 1); 2526 } 2527 else 2528 { 2529 pResMat->PutString(ScGlobal::GetRscString(STR_NV_STR), 1, 1); 2530 } 2531 2532 double fR2 = fSSreg / (fSSreg + fSSresid); 2533 pResMat->PutDouble(fR2, 0, 2); 2534 } 2535 } 2536 PushMatrix(pResMat); 2537 } 2538 else // calculate multiple regression; 2539 { 2540 // Uses a QR decomposition X = QR. The solution B = (X'X)^(-1) * X' * Y 2541 // becomes B = R^(-1) * Q' * Y 2542 if (nCase ==2) // Y is column 2543 { 2544 ::std::vector< double> aVecR(N); // for QR decomposition 2545 // Enough memory for needed matrices? 2546 ScMatrixRef pMeans = GetNewMat(K, 1); // mean of each column 2547 ScMatrixRef pMatZ; // for Q' * Y , inter alia 2548 if (bStats) 2549 pMatZ = pMatY->Clone(); // Y is used in statistic, keep it 2550 else 2551 pMatZ = pMatY; // Y can be overwritten 2552 ScMatrixRef pSlopes = GetNewMat(1,K); // from b1 to bK 2553 if (!pMeans || !pMatZ || !pSlopes) 2554 { 2555 PushError(errCodeOverflow); 2556 return; 2557 } 2558 if (bConstant) 2559 { 2560 lcl_CalculateColumnMeans(pMatX, pMeans, K, N); 2561 lcl_CalculateColumnsDelta(pMatX, pMeans, K, N); 2562 } 2563 if (!lcl_CalculateQRdecomposition(pMatX, aVecR, K, N)) 2564 { 2565 PushNoValue(); 2566 return; 2567 } 2568 // Later on we will divide by elements of aVecR, so make sure 2569 // that they aren't zero. 2570 bool bIsSingular=false; 2571 for (SCSIZE row=0; row < K && !bIsSingular; row++) 2572 bIsSingular = bIsSingular || aVecR[row]==0.0; 2573 if (bIsSingular) 2574 { 2575 PushNoValue(); 2576 return; 2577 } 2578 // Z = Q' Y; 2579 for (SCSIZE col = 0; col < K; col++) 2580 { 2581 lcl_ApplyHouseholderTransformation(pMatX, col, pMatZ, N); 2582 } 2583 // B = R^(-1) * Q' * Y <=> B = R^(-1) * Z <=> R * B = Z 2584 // result Z should have zeros for index>=K; if not, ignore values 2585 for (SCSIZE col = 0; col < K ; col++) 2586 { 2587 pSlopes->PutDouble( pMatZ->GetDouble(col), col); 2588 } 2589 lcl_SolveWithUpperRightTriangle(pMatX, aVecR, pSlopes, K, false); 2590 double fIntercept = 0.0; 2591 if (bConstant) 2592 fIntercept = fMeanY - lcl_GetSumProduct(pMeans,pSlopes,K); 2593 // Fill first line in result matrix 2594 pResMat->PutDouble(_bRKP ? exp(fIntercept) : fIntercept, K, 0 ); 2595 for (SCSIZE i = 0; i < K; i++) 2596 pResMat->PutDouble(_bRKP ? exp(pSlopes->GetDouble(i)) 2597 : pSlopes->GetDouble(i) , K-1-i, 0); 2598 2599 2600 if (bStats) 2601 { 2602 double fSSreg = 0.0; 2603 double fSSresid = 0.0; 2604 // re-use memory of Z; 2605 pMatZ->FillDouble(0.0, 0, 0, 0, N-1); 2606 // Z = R * Slopes 2607 lcl_ApplyUpperRightTriangle(pMatX, aVecR, pSlopes, pMatZ, K, false); 2608 // Z = Q * Z, that is Q * R * Slopes = X * Slopes 2609 for (SCSIZE colp1 = K; colp1 > 0; colp1--) 2610 { 2611 lcl_ApplyHouseholderTransformation(pMatX, colp1-1, pMatZ,N); 2612 } 2613 fSSreg =lcl_GetSumProduct(pMatZ, pMatZ, N); 2614 // re-use Y for residuals, Y = Y-Z 2615 for (SCSIZE row = 0; row < N; row++) 2616 pMatY->PutDouble(pMatY->GetDouble(row) - pMatZ->GetDouble(row), row); 2617 fSSresid = lcl_GetSumProduct(pMatY, pMatY, N); 2618 pResMat->PutDouble(fSSreg, 0, 4); 2619 pResMat->PutDouble(fSSresid, 1, 4); 2620 2621 double fDegreesFreedom =static_cast<double>( (bConstant) ? N-K-1 : N-K ); 2622 pResMat->PutDouble(fDegreesFreedom, 1, 3); 2623 2624 if (fDegreesFreedom == 0.0 || fSSresid == 0.0 || fSSreg == 0.0) 2625 { // exact fit; incl. observed values Y are identical 2626 pResMat->PutDouble(0.0, 1, 4); // SSresid 2627 // F = (SSreg/K) / (SSresid/df) = #DIV/0! 2628 pResMat->PutString(ScGlobal::GetRscString(STR_NV_STR), 0, 3); // F 2629 // RMSE = sqrt(SSresid / df) = sqrt(0 / df) = 0 2630 pResMat->PutDouble(0.0, 1, 2); // RMSE 2631 // SigmaSlope[i] = RMSE * sqrt(matrix[i,i]) = 0 * sqrt(...) = 0 2632 for (SCSIZE i=0; i<K; i++) 2633 pResMat->PutDouble(0.0, K-1-i, 1); 2634 2635 // SigmaIntercept = RMSE * sqrt(...) = 0 2636 if (bConstant) 2637 pResMat->PutDouble(0.0, K, 1); //SigmaIntercept 2638 else 2639 pResMat->PutString(ScGlobal::GetRscString(STR_NV_STR), K, 1); 2640 2641 // R^2 = SSreg / (SSreg + SSresid) = 1.0 2642 pResMat->PutDouble(1.0, 0, 2); // R^2 2643 } 2644 else 2645 { 2646 double fFstatistic = (fSSreg / static_cast<double>(K)) 2647 / (fSSresid / fDegreesFreedom); 2648 pResMat->PutDouble(fFstatistic, 0, 3); 2649 2650 // standard error of estimate = root mean SSE 2651 double fRMSE = sqrt(fSSresid / fDegreesFreedom); 2652 pResMat->PutDouble(fRMSE, 1, 2); 2653 2654 // standard error of slopes 2655 // = RMSE * sqrt(diagonal element of (R' R)^(-1) ) 2656 // standard error of intercept 2657 // = RMSE * sqrt( Xmean * (R' R)^(-1) * Xmean' + 1/N) 2658 // (R' R)^(-1) = R^(-1) * (R')^(-1). Do not calculate it as 2659 // a whole matrix, but iterate over unit vectors. 2660 double fSigmaSlope = 0.0; 2661 double fSigmaIntercept = 0.0; 2662 double fPart; // for Xmean * single column of (R' R)^(-1) 2663 for (SCSIZE col = 0; col < K; col++) 2664 { 2665 //re-use memory of MatZ 2666 pMatZ->FillDouble(0.0,0,0,0,K-1); // Z = unit vector e 2667 pMatZ->PutDouble(1.0, col); 2668 //Solve R' * Z = e 2669 lcl_SolveWithLowerLeftTriangle(pMatX, aVecR, pMatZ, K, false); 2670 // Solve R * Znew = Zold 2671 lcl_SolveWithUpperRightTriangle(pMatX, aVecR, pMatZ, K, false); 2672 // now Z is column col in (R' R)^(-1) 2673 fSigmaSlope = fRMSE * sqrt(pMatZ->GetDouble(col)); 2674 pResMat->PutDouble(fSigmaSlope, K-1-col, 1); 2675 // (R' R) ^(-1) is symmetric 2676 if (bConstant) 2677 { 2678 fPart = lcl_GetSumProduct(pMeans, pMatZ, K); 2679 fSigmaIntercept += fPart * pMeans->GetDouble(col); 2680 } 2681 } 2682 if (bConstant) 2683 { 2684 fSigmaIntercept = fRMSE 2685 * sqrt(fSigmaIntercept + 1.0 / static_cast<double>(N)); 2686 pResMat->PutDouble(fSigmaIntercept, K, 1); 2687 } 2688 else 2689 { 2690 pResMat->PutString(ScGlobal::GetRscString(STR_NV_STR), K, 1); 2691 } 2692 2693 double fR2 = fSSreg / (fSSreg + fSSresid); 2694 pResMat->PutDouble(fR2, 0, 2); 2695 } 2696 } 2697 PushMatrix(pResMat); 2698 } 2699 else // nCase == 3, Y is row, all matrices are transposed 2700 { 2701 ::std::vector< double> aVecR(N); // for QR decomposition 2702 // Enough memory for needed matrices? 2703 ScMatrixRef pMeans = GetNewMat(1, K); // mean of each row 2704 ScMatrixRef pMatZ; // for Q' * Y , inter alia 2705 if (bStats) 2706 pMatZ = pMatY->Clone(); // Y is used in statistic, keep it 2707 else 2708 pMatZ = pMatY; // Y can be overwritten 2709 ScMatrixRef pSlopes = GetNewMat(K,1); // from b1 to bK 2710 if (!pMeans || !pMatZ || !pSlopes) 2711 { 2712 PushError(errCodeOverflow); 2713 return; 2714 } 2715 if (bConstant) 2716 { 2717 lcl_CalculateRowMeans(pMatX, pMeans, N, K); 2718 lcl_CalculateRowsDelta(pMatX, pMeans, N, K); 2719 } 2720 2721 if (!lcl_TCalculateQRdecomposition(pMatX, aVecR, K, N)) 2722 { 2723 PushNoValue(); 2724 return; 2725 } 2726 2727 // Later on we will divide by elements of aVecR, so make sure 2728 // that they aren't zero. 2729 bool bIsSingular=false; 2730 for (SCSIZE row=0; row < K && !bIsSingular; row++) 2731 bIsSingular = bIsSingular || aVecR[row]==0.0; 2732 if (bIsSingular) 2733 { 2734 PushNoValue(); 2735 return; 2736 } 2737 // Z = Q' Y 2738 for (SCSIZE row = 0; row < K; row++) 2739 { 2740 lcl_TApplyHouseholderTransformation(pMatX, row, pMatZ, N); 2741 } 2742 // B = R^(-1) * Q' * Y <=> B = R^(-1) * Z <=> R * B = Z 2743 // result Z should have zeros for index>=K; if not, ignore values 2744 for (SCSIZE col = 0; col < K ; col++) 2745 { 2746 pSlopes->PutDouble( pMatZ->GetDouble(col), col); 2747 } 2748 lcl_SolveWithUpperRightTriangle(pMatX, aVecR, pSlopes, K, true); 2749 double fIntercept = 0.0; 2750 if (bConstant) 2751 fIntercept = fMeanY - lcl_GetSumProduct(pMeans,pSlopes,K); 2752 // Fill first line in result matrix 2753 pResMat->PutDouble(_bRKP ? exp(fIntercept) : fIntercept, K, 0 ); 2754 for (SCSIZE i = 0; i < K; i++) 2755 pResMat->PutDouble(_bRKP ? exp(pSlopes->GetDouble(i)) 2756 : pSlopes->GetDouble(i) , K-1-i, 0); 2757 2758 2759 if (bStats) 2760 { 2761 double fSSreg = 0.0; 2762 double fSSresid = 0.0; 2763 // re-use memory of Z; 2764 pMatZ->FillDouble(0.0, 0, 0, N-1, 0); 2765 // Z = R * Slopes 2766 lcl_ApplyUpperRightTriangle(pMatX, aVecR, pSlopes, pMatZ, K, true); 2767 // Z = Q * Z, that is Q * R * Slopes = X * Slopes 2768 for (SCSIZE rowp1 = K; rowp1 > 0; rowp1--) 2769 { 2770 lcl_TApplyHouseholderTransformation(pMatX, rowp1-1, pMatZ,N); 2771 } 2772 fSSreg =lcl_GetSumProduct(pMatZ, pMatZ, N); 2773 // re-use Y for residuals, Y = Y-Z 2774 for (SCSIZE col = 0; col < N; col++) 2775 pMatY->PutDouble(pMatY->GetDouble(col) - pMatZ->GetDouble(col), col); 2776 fSSresid = lcl_GetSumProduct(pMatY, pMatY, N); 2777 pResMat->PutDouble(fSSreg, 0, 4); 2778 pResMat->PutDouble(fSSresid, 1, 4); 2779 2780 double fDegreesFreedom =static_cast<double>( (bConstant) ? N-K-1 : N-K ); 2781 pResMat->PutDouble(fDegreesFreedom, 1, 3); 2782 2783 if (fDegreesFreedom == 0.0 || fSSresid == 0.0 || fSSreg == 0.0) 2784 { // exact fit; incl. case observed values Y are identical 2785 pResMat->PutDouble(0.0, 1, 4); // SSresid 2786 // F = (SSreg/K) / (SSresid/df) = #DIV/0! 2787 pResMat->PutString(ScGlobal::GetRscString(STR_NV_STR), 0, 3); // F 2788 // RMSE = sqrt(SSresid / df) = sqrt(0 / df) = 0 2789 pResMat->PutDouble(0.0, 1, 2); // RMSE 2790 // SigmaSlope[i] = RMSE * sqrt(matrix[i,i]) = 0 * sqrt(...) = 0 2791 for (SCSIZE i=0; i<K; i++) 2792 pResMat->PutDouble(0.0, K-1-i, 1); 2793 2794 // SigmaIntercept = RMSE * sqrt(...) = 0 2795 if (bConstant) 2796 pResMat->PutDouble(0.0, K, 1); //SigmaIntercept 2797 else 2798 pResMat->PutString(ScGlobal::GetRscString(STR_NV_STR), K, 1); 2799 2800 // R^2 = SSreg / (SSreg + SSresid) = 1.0 2801 pResMat->PutDouble(1.0, 0, 2); // R^2 2802 } 2803 else 2804 { 2805 double fFstatistic = (fSSreg / static_cast<double>(K)) 2806 / (fSSresid / fDegreesFreedom); 2807 pResMat->PutDouble(fFstatistic, 0, 3); 2808 2809 // standard error of estimate = root mean SSE 2810 double fRMSE = sqrt(fSSresid / fDegreesFreedom); 2811 pResMat->PutDouble(fRMSE, 1, 2); 2812 2813 // standard error of slopes 2814 // = RMSE * sqrt(diagonal element of (R' R)^(-1) ) 2815 // standard error of intercept 2816 // = RMSE * sqrt( Xmean * (R' R)^(-1) * Xmean' + 1/N) 2817 // (R' R)^(-1) = R^(-1) * (R')^(-1). Do not calculate it as 2818 // a whole matrix, but iterate over unit vectors. 2819 // (R' R) ^(-1) is symmetric 2820 double fSigmaSlope = 0.0; 2821 double fSigmaIntercept = 0.0; 2822 double fPart; // for Xmean * single col of (R' R)^(-1) 2823 for (SCSIZE row = 0; row < K; row++) 2824 { 2825 //re-use memory of MatZ 2826 pMatZ->FillDouble(0.0,0,0,K-1,0); // Z = unit vector e 2827 pMatZ->PutDouble(1.0, row); 2828 //Solve R' * Z = e 2829 lcl_SolveWithLowerLeftTriangle(pMatX, aVecR, pMatZ, K, true); 2830 // Solve R * Znew = Zold 2831 lcl_SolveWithUpperRightTriangle(pMatX, aVecR, pMatZ, K, true); 2832 // now Z is column col in (R' R)^(-1) 2833 fSigmaSlope = fRMSE * sqrt(pMatZ->GetDouble(row)); 2834 pResMat->PutDouble(fSigmaSlope, K-1-row, 1); 2835 if (bConstant) 2836 { 2837 fPart = lcl_GetSumProduct(pMeans, pMatZ, K); 2838 fSigmaIntercept += fPart * pMeans->GetDouble(row); 2839 } 2840 } 2841 if (bConstant) 2842 { 2843 fSigmaIntercept = fRMSE 2844 * sqrt(fSigmaIntercept + 1.0 / static_cast<double>(N)); 2845 pResMat->PutDouble(fSigmaIntercept, K, 1); 2846 } 2847 else 2848 { 2849 pResMat->PutString(ScGlobal::GetRscString(STR_NV_STR), K, 1); 2850 } 2851 2852 double fR2 = fSSreg / (fSSreg + fSSresid); 2853 pResMat->PutDouble(fR2, 0, 2); 2854 } 2855 } 2856 PushMatrix(pResMat); 2857 } 2858 } 2859 return; 2860 } 2861 2862 void ScInterpreter::ScTrend() 2863 { 2864 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "sc", "er", "ScInterpreter::ScTrend" ); 2865 CalculateTrendGrowth(false); 2866 } 2867 2868 void ScInterpreter::ScGrowth() 2869 { 2870 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "sc", "er", "ScInterpreter::ScGrowth" ); 2871 CalculateTrendGrowth(true); 2872 } 2873 2874 void ScInterpreter::CalculateTrendGrowth(bool _bGrowth) 2875 { 2876 sal_uInt8 nParamCount = GetByte(); 2877 if ( !MustHaveParamCount( nParamCount, 1, 4 ) ) 2878 return; 2879 2880 // optional fourth parameter 2881 bool bConstant; 2882 if (nParamCount == 4) 2883 bConstant = GetBool(); 2884 else 2885 bConstant = true; 2886 2887 // The third parameter may be missing in ODF, although the fourth parameter 2888 // is present. Default values depend on data not yet read. 2889 ScMatrixRef pMatNewX; 2890 if (nParamCount >= 3) 2891 { 2892 if (IsMissing()) 2893 { 2894 Pop(); 2895 pMatNewX = NULL; 2896 } 2897 else 2898 pMatNewX = GetMatrix(); 2899 } 2900 else 2901 pMatNewX = NULL; 2902 2903 // In ODF1.2 empty second parameter (which is two ;; ) is allowed. 2904 // Defaults will be set in CheckMatrix. 2905 ScMatrixRef pMatX; 2906 if (nParamCount >= 2) 2907 { 2908 if (IsMissing()) 2909 { 2910 Pop(); 2911 pMatX = NULL; 2912 } 2913 else 2914 { 2915 pMatX = GetMatrix(); 2916 } 2917 } 2918 else 2919 pMatX = NULL; 2920 2921 ScMatrixRef pMatY; 2922 pMatY = GetMatrix(); 2923 if (!pMatY) 2924 { 2925 PushIllegalParameter(); 2926 return; 2927 } 2928 2929 // 1 = simple; 2 = multiple with Y as column; 3 = multiple with Y as row 2930 sal_uInt8 nCase; 2931 2932 SCSIZE nCX, nCY; // number of columns 2933 SCSIZE nRX, nRY; // number of rows 2934 SCSIZE K = 0, N = 0; // K=number of variables X, N=number of data samples 2935 if ( !CheckMatrix(_bGrowth,nCase,nCX,nCY,nRX,nRY,K,N,pMatX,pMatY) ) 2936 { 2937 PushIllegalParameter(); 2938 return; 2939 } 2940 2941 // Enough data samples? 2942 if ( (bConstant && (N<K+1)) || (!bConstant && (N<K)) || (N<1) || (K<1) ) 2943 { 2944 PushIllegalParameter(); 2945 return; 2946 } 2947 2948 // Set default pMatNewX if necessary 2949 SCSIZE nCXN, nRXN; 2950 SCSIZE nCountXN; 2951 if (!pMatNewX) 2952 { 2953 nCXN = nCX; 2954 nRXN = nRX; 2955 nCountXN = nCXN * nRXN; 2956 pMatNewX = pMatX->Clone(); // pMatX will be changed to X-meanX 2957 } 2958 else 2959 { 2960 pMatNewX->GetDimensions(nCXN, nRXN); 2961 if ((nCase == 2 && K != nCXN) || (nCase == 3 && K != nRXN)) 2962 { 2963 PushIllegalArgument(); 2964 return; 2965 } 2966 nCountXN = nCXN * nRXN; 2967 for ( SCSIZE i = 0; i < nCountXN; i++ ) 2968 if (!pMatNewX->IsValue(i)) 2969 { 2970 PushIllegalArgument(); 2971 return; 2972 } 2973 } 2974 ScMatrixRef pResMat; // size depends on nCase 2975 if (nCase == 1) 2976 pResMat = GetNewMat(nCXN,nRXN); 2977 else 2978 { 2979 if (nCase==2) 2980 pResMat = GetNewMat(1,nRXN); 2981 else 2982 pResMat = GetNewMat(nCXN,1); 2983 } 2984 if (!pResMat) 2985 { 2986 PushError(errCodeOverflow); 2987 return; 2988 } 2989 // Uses sum(x-MeanX)^2 and not [sum x^2]-N * MeanX^2 in case bConstant. 2990 // Clone constant matrices, so that Mat = Mat - Mean is possible. 2991 double fMeanY = 0.0; 2992 if (bConstant) 2993 { 2994 ScMatrixRef pCopyX = pMatX->CloneIfConst(); 2995 ScMatrixRef pCopyY = pMatY->CloneIfConst(); 2996 if (!pCopyX || !pCopyY) 2997 { 2998 PushError(errStackOverflow); 2999 return; 3000 } 3001 pMatX = pCopyX; 3002 pMatY = pCopyY; 3003 // DeltaY is possible here; DeltaX depends on nCase, so later 3004 fMeanY = lcl_GetMeanOverAll(pMatY, N); 3005 for (SCSIZE i=0; i<N; i++) 3006 { 3007 pMatY->PutDouble( ::rtl::math::approxSub(pMatY->GetDouble(i),fMeanY), i ); 3008 } 3009 } 3010 3011 if (nCase==1) 3012 { 3013 // calculate simple regression 3014 double fMeanX = 0.0; 3015 if (bConstant) 3016 { // Mat = Mat - Mean 3017 fMeanX = lcl_GetMeanOverAll(pMatX, N); 3018 for (SCSIZE i=0; i<N; i++) 3019 { 3020 pMatX->PutDouble( ::rtl::math::approxSub(pMatX->GetDouble(i),fMeanX), i ); 3021 } 3022 } 3023 double fSumXY = lcl_GetSumProduct(pMatX,pMatY,N); 3024 double fSumX2 = lcl_GetSumProduct(pMatX,pMatX,N); 3025 if (fSumX2==0.0) 3026 { 3027 PushNoValue(); // all x-values are identical 3028 return; 3029 } 3030 double fSlope = fSumXY / fSumX2; 3031 double fIntercept = 0.0; 3032 double fHelp; 3033 if (bConstant) 3034 { 3035 fIntercept = fMeanY - fSlope * fMeanX; 3036 for (SCSIZE i = 0; i < nCountXN; i++) 3037 { 3038 fHelp = pMatNewX->GetDouble(i)*fSlope + fIntercept; 3039 pResMat->PutDouble(_bGrowth ? exp(fHelp) : fHelp, i); 3040 } 3041 } 3042 else 3043 { 3044 for (SCSIZE i = 0; i < nCountXN; i++) 3045 { 3046 fHelp = pMatNewX->GetDouble(i)*fSlope; 3047 pResMat->PutDouble(_bGrowth ? exp(fHelp) : fHelp, i); 3048 } 3049 } 3050 } 3051 else // calculate multiple regression; 3052 { 3053 if (nCase ==2) // Y is column 3054 { 3055 ::std::vector< double> aVecR(N); // for QR decomposition 3056 // Enough memory for needed matrices? 3057 ScMatrixRef pMeans = GetNewMat(K, 1); // mean of each column 3058 ScMatrixRef pSlopes = GetNewMat(1,K); // from b1 to bK 3059 if (!pMeans || !pSlopes) 3060 { 3061 PushError(errCodeOverflow); 3062 return; 3063 } 3064 if (bConstant) 3065 { 3066 lcl_CalculateColumnMeans(pMatX, pMeans, K, N); 3067 lcl_CalculateColumnsDelta(pMatX, pMeans, K, N); 3068 } 3069 if (!lcl_CalculateQRdecomposition(pMatX, aVecR, K, N)) 3070 { 3071 PushNoValue(); 3072 return; 3073 } 3074 // Later on we will divide by elements of aVecR, so make sure 3075 // that they aren't zero. 3076 bool bIsSingular=false; 3077 for (SCSIZE row=0; row < K && !bIsSingular; row++) 3078 bIsSingular = bIsSingular || aVecR[row]==0.0; 3079 if (bIsSingular) 3080 { 3081 PushNoValue(); 3082 return; 3083 } 3084 // Z := Q' Y; Y is overwritten with result Z 3085 for (SCSIZE col = 0; col < K; col++) 3086 { 3087 lcl_ApplyHouseholderTransformation(pMatX, col, pMatY, N); 3088 } 3089 // B = R^(-1) * Q' * Y <=> B = R^(-1) * Z <=> R * B = Z 3090 // result Z should have zeros for index>=K; if not, ignore values 3091 for (SCSIZE col = 0; col < K ; col++) 3092 { 3093 pSlopes->PutDouble( pMatY->GetDouble(col), col); 3094 } 3095 lcl_SolveWithUpperRightTriangle(pMatX, aVecR, pSlopes, K, false); 3096 3097 // Fill result matrix 3098 lcl_MFastMult(pMatNewX,pSlopes,pResMat,nRXN,K,1); 3099 if (bConstant) 3100 { 3101 double fIntercept = fMeanY - lcl_GetSumProduct(pMeans,pSlopes,K); 3102 for (SCSIZE row = 0; row < nRXN; row++) 3103 pResMat->PutDouble(pResMat->GetDouble(row)+fIntercept, row); 3104 } 3105 if (_bGrowth) 3106 { 3107 for (SCSIZE i = 0; i < nRXN; i++) 3108 pResMat->PutDouble(exp(pResMat->GetDouble(i)), i); 3109 } 3110 } 3111 else 3112 { // nCase == 3, Y is row, all matrices are transposed 3113 3114 ::std::vector< double> aVecR(N); // for QR decomposition 3115 // Enough memory for needed matrices? 3116 ScMatrixRef pMeans = GetNewMat(1, K); // mean of each row 3117 ScMatrixRef pSlopes = GetNewMat(K,1); // row from b1 to bK 3118 if (!pMeans || !pSlopes) 3119 { 3120 PushError(errCodeOverflow); 3121 return; 3122 } 3123 if (bConstant) 3124 { 3125 lcl_CalculateRowMeans(pMatX, pMeans, N, K); 3126 lcl_CalculateRowsDelta(pMatX, pMeans, N, K); 3127 } 3128 if (!lcl_TCalculateQRdecomposition(pMatX, aVecR, K, N)) 3129 { 3130 PushNoValue(); 3131 return; 3132 } 3133 // Later on we will divide by elements of aVecR, so make sure 3134 // that they aren't zero. 3135 bool bIsSingular=false; 3136 for (SCSIZE row=0; row < K && !bIsSingular; row++) 3137 bIsSingular = bIsSingular || aVecR[row]==0.0; 3138 if (bIsSingular) 3139 { 3140 PushNoValue(); 3141 return; 3142 } 3143 // Z := Q' Y; Y is overwritten with result Z 3144 for (SCSIZE row = 0; row < K; row++) 3145 { 3146 lcl_TApplyHouseholderTransformation(pMatX, row, pMatY, N); 3147 } 3148 // B = R^(-1) * Q' * Y <=> B = R^(-1) * Z <=> R * B = Z 3149 // result Z should have zeros for index>=K; if not, ignore values 3150 for (SCSIZE col = 0; col < K ; col++) 3151 { 3152 pSlopes->PutDouble( pMatY->GetDouble(col), col); 3153 } 3154 lcl_SolveWithUpperRightTriangle(pMatX, aVecR, pSlopes, K, true); 3155 3156 // Fill result matrix 3157 lcl_MFastMult(pSlopes,pMatNewX,pResMat,1,K,nCXN); 3158 if (bConstant) 3159 { 3160 double fIntercept = fMeanY - lcl_GetSumProduct(pMeans,pSlopes,K); 3161 for (SCSIZE col = 0; col < nCXN; col++) 3162 pResMat->PutDouble(pResMat->GetDouble(col)+fIntercept, col); 3163 } 3164 if (_bGrowth) 3165 { 3166 for (SCSIZE i = 0; i < nCXN; i++) 3167 pResMat->PutDouble(exp(pResMat->GetDouble(i)), i); 3168 } 3169 } 3170 } 3171 PushMatrix(pResMat); 3172 return; 3173 } 3174 3175 void ScInterpreter::ScMatRef() 3176 { 3177 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "sc", "er", "ScInterpreter::ScMatRef" ); 3178 // Falls Deltarefs drin sind... 3179 Push( (FormulaToken&)*pCur ); 3180 ScAddress aAdr; 3181 PopSingleRef( aAdr ); 3182 ScFormulaCell* pCell = (ScFormulaCell*) GetCell( aAdr ); 3183 if( pCell && pCell->GetCellType() == CELLTYPE_FORMULA ) 3184 { 3185 const ScMatrix* pMat = pCell->GetMatrix(); 3186 if( pMat ) 3187 { 3188 SCSIZE nCols, nRows; 3189 pMat->GetDimensions( nCols, nRows ); 3190 SCSIZE nC = static_cast<SCSIZE>(aPos.Col() - aAdr.Col()); 3191 SCSIZE nR = static_cast<SCSIZE>(aPos.Row() - aAdr.Row()); 3192 if ((nCols <= nC && nCols != 1) || (nRows <= nR && nRows != 1)) 3193 PushNA(); 3194 else 3195 { 3196 ScMatValType nMatValType; 3197 const ScMatrixValue* pMatVal = pMat->Get( nC, nR, nMatValType); 3198 if (ScMatrix::IsNonValueType( nMatValType)) 3199 { 3200 if (ScMatrix::IsEmptyPathType( nMatValType)) 3201 { // result of empty sal_False jump path 3202 nFuncFmtType = NUMBERFORMAT_LOGICAL; 3203 PushInt(0); 3204 } 3205 else if (ScMatrix::IsEmptyType( nMatValType)) 3206 { 3207 // Not inherited (really?) and display as empty string, not 0. 3208 PushTempToken( new ScEmptyCellToken( false, true)); 3209 } 3210 else 3211 PushString( pMatVal->GetString() ); 3212 } 3213 else 3214 { 3215 PushDouble(pMatVal->fVal); // handles DoubleError 3216 pDok->GetNumberFormatInfo( nCurFmtType, nCurFmtIndex, aAdr, pCell ); 3217 nFuncFmtType = nCurFmtType; 3218 nFuncFmtIndex = nCurFmtIndex; 3219 } 3220 } 3221 } 3222 else 3223 { 3224 // If not a result matrix, obtain the cell value. 3225 sal_uInt16 nErr = pCell->GetErrCode(); 3226 if (nErr) 3227 PushError( nErr ); 3228 else if( pCell->IsValue() ) 3229 PushDouble( pCell->GetValue() ); 3230 else 3231 { 3232 String aVal; 3233 pCell->GetString( aVal ); 3234 PushString( aVal ); 3235 } 3236 pDok->GetNumberFormatInfo( nCurFmtType, nCurFmtIndex, aAdr, pCell ); 3237 nFuncFmtType = nCurFmtType; 3238 nFuncFmtIndex = nCurFmtIndex; 3239 } 3240 } 3241 else 3242 PushError( errNoRef ); 3243 } 3244 3245 void ScInterpreter::ScInfo() 3246 { 3247 RTL_LOGFILE_CONTEXT_AUTHOR( aLogger, "sc", "er", "ScInterpreter::ScInfo" ); 3248 if( MustHaveParamCount( GetByte(), 1 ) ) 3249 { 3250 String aStr = GetString(); 3251 ScCellKeywordTranslator::transKeyword(aStr, ScGlobal::GetLocale(), ocInfo); 3252 if( aStr.EqualsAscii( "SYSTEM" ) ) 3253 PushString( String( RTL_CONSTASCII_USTRINGPARAM( SC_INFO_OSVERSION ) ) ); 3254 else if( aStr.EqualsAscii( "OSVERSION" ) ) 3255 PushString( String( RTL_CONSTASCII_USTRINGPARAM( "Windows (32-bit) NT 5.01" ) ) ); 3256 else if( aStr.EqualsAscii( "RELEASE" ) ) 3257 PushString( ::utl::Bootstrap::getBuildIdData( ::rtl::OUString() ) ); 3258 else if( aStr.EqualsAscii( "NUMFILE" ) ) 3259 PushDouble( 1 ); 3260 else if( aStr.EqualsAscii( "RECALC" ) ) 3261 PushString( ScGlobal::GetRscString( pDok->GetAutoCalc() ? STR_RECALC_AUTO : STR_RECALC_MANUAL ) ); 3262 else 3263 PushIllegalArgument(); 3264 } 3265 } 3266