244 lines
9.0 KiB
C++
244 lines
9.0 KiB
C++
/*
|
|
* Legal Notice
|
|
*
|
|
* This document and associated source code (the "Work") is a part of a
|
|
* benchmark specification maintained by the TPC.
|
|
*
|
|
* The TPC reserves all right, title, and interest to the Work as provided
|
|
* under U.S. and international laws, including without limitation all patent
|
|
* and trademark rights therein.
|
|
*
|
|
* No Warranty
|
|
*
|
|
* 1.1 TO THE MAXIMUM EXTENT PERMITTED BY APPLICABLE LAW, THE INFORMATION
|
|
* CONTAINED HEREIN IS PROVIDED "AS IS" AND WITH ALL FAULTS, AND THE
|
|
* AUTHORS AND DEVELOPERS OF THE WORK HEREBY DISCLAIM ALL OTHER
|
|
* WARRANTIES AND CONDITIONS, EITHER EXPRESS, IMPLIED OR STATUTORY,
|
|
* INCLUDING, BUT NOT LIMITED TO, ANY (IF ANY) IMPLIED WARRANTIES,
|
|
* DUTIES OR CONDITIONS OF MERCHANTABILITY, OF FITNESS FOR A PARTICULAR
|
|
* PURPOSE, OF ACCURACY OR COMPLETENESS OF RESPONSES, OF RESULTS, OF
|
|
* WORKMANLIKE EFFORT, OF LACK OF VIRUSES, AND OF LACK OF NEGLIGENCE.
|
|
* ALSO, THERE IS NO WARRANTY OR CONDITION OF TITLE, QUIET ENJOYMENT,
|
|
* QUIET POSSESSION, CORRESPONDENCE TO DESCRIPTION OR NON-INFRINGEMENT
|
|
* WITH REGARD TO THE WORK.
|
|
* 1.2 IN NO EVENT WILL ANY AUTHOR OR DEVELOPER OF THE WORK BE LIABLE TO
|
|
* ANY OTHER PARTY FOR ANY DAMAGES, INCLUDING BUT NOT LIMITED TO THE
|
|
* COST OF PROCURING SUBSTITUTE GOODS OR SERVICES, LOST PROFITS, LOSS
|
|
* OF USE, LOSS OF DATA, OR ANY INCIDENTAL, CONSEQUENTIAL, DIRECT,
|
|
* INDIRECT, OR SPECIAL DAMAGES WHETHER UNDER CONTRACT, TORT, WARRANTY,
|
|
* OR OTHERWISE, ARISING IN ANY WAY OUT OF THIS OR ANY OTHER AGREEMENT
|
|
* RELATING TO THE WORK, WHETHER OR NOT SUCH AUTHOR OR DEVELOPER HAD
|
|
* ADVANCE NOTICE OF THE POSSIBILITY OF SUCH DAMAGES.
|
|
*
|
|
* Contributors
|
|
* - Sergey Vasilevskiy, Cecil Reames, Matt Emmerton, Doug Johnson
|
|
*/
|
|
|
|
/*
|
|
* Class representing the Financial table.
|
|
*/
|
|
#ifndef FINANCIAL_TABLE_H
|
|
#define FINANCIAL_TABLE_H
|
|
|
|
#include "EGenTables_common.h"
|
|
#include "CompanyTable.h"
|
|
#include "utilities/Money.h"
|
|
|
|
#include "input/DataFileManager.h"
|
|
|
|
namespace TPCE {
|
|
|
|
const int iYearsForFins = 5;
|
|
const int iQuartersInYear = 4;
|
|
const int iFinsPerCompany = iYearsForFins * iQuartersInYear; // 5 years of 4 quaters each year
|
|
|
|
// multiplier to get the diluted number of shares from outstanding
|
|
const double fDilutedSharesMultiplier = 1.1;
|
|
|
|
// Multipliers for previous quarter to get the current quarter data
|
|
const double fFinDataDownMult = 0.9;
|
|
const double fFinDataUpMult = 1.15;
|
|
const double fFinDataIncr = 0.00000000000001;
|
|
|
|
const double fFinancialRevenueMin = 100000.00;
|
|
const double fFinancialRevenueMax = 16000000000.00;
|
|
|
|
const double fFinancialEarningsMin = -300000000.00;
|
|
const double fFinancialEarningsMax = 3000000000.00;
|
|
|
|
const INT64 iFinancialOutBasicMin = 400000;
|
|
const INT64 iFinancialOutBasicMax = INT64_CONST(9500000000);
|
|
|
|
const double fFinancialInventMin = 0.00;
|
|
const double fFinancialInventMax = 2000000000.00;
|
|
|
|
const double fFinancialAssetsMin = 100000.00;
|
|
const double fFinancialAssetsMax = 65000000000.00;
|
|
|
|
const double fFinancialLiabMin = 100000.00;
|
|
const double fFinancialLiabMax = 35000000000.00;
|
|
|
|
// Number of RNG calls to skip for one row in order
|
|
// to not use any of the random values from the previous row.
|
|
//
|
|
const int iRNGSkipOneRowFinancial = 6 + iFinsPerCompany * 6;
|
|
|
|
typedef struct FINANCIAL_GEN_ROW {
|
|
FINANCIAL_ROW m_financials[iFinsPerCompany];
|
|
} * PFINANCIAL_GEN_ROW;
|
|
|
|
class CFinancialTable : public TableTemplate<FINANCIAL_GEN_ROW> {
|
|
CCompanyTable m_CompanyTable;
|
|
int m_iFinYear; // first year to generate financials
|
|
int m_iFinQuarter; // first quarter to generate financials (0-based)
|
|
// Number of times GenerateNextRecord() was called for the current company
|
|
// data. Needed to decide when to generate next company's data.
|
|
int m_iRowsGeneratedPerCompany;
|
|
// Stores whether there is another company(s) for which to
|
|
// generate financial data.
|
|
bool m_bMoreCompanies;
|
|
TIdent m_iFinancialCountForOneLoadUnit;
|
|
|
|
//
|
|
// Generate the financial data for the next company.
|
|
//
|
|
// Return whether there are more companies to generate data for.
|
|
//
|
|
bool GenerateFinancialRows() {
|
|
TIdent FI_CO_ID;
|
|
int iFinYear, iFinQuarter;
|
|
int i;
|
|
CMoney fRev, fEarn, fInvent, fAssets, fLiab, fBasicEPS, fDilutEPS, fMargin;
|
|
INT64 iOutBasic, iOutDilut;
|
|
|
|
// Set starting values for financial values
|
|
FI_CO_ID = m_CompanyTable.GetCurrentCO_ID();
|
|
iFinYear = m_iFinYear;
|
|
iFinQuarter = m_iFinQuarter;
|
|
|
|
fRev = m_rnd.RndDoubleIncrRange(fFinancialRevenueMin, fFinancialRevenueMax, 0.01);
|
|
fEarn = m_rnd.RndDoubleIncrRange(
|
|
fFinancialEarningsMin, fRev < fFinancialEarningsMax ? fRev.DollarAmount() : fFinancialEarningsMax, 0.01);
|
|
iOutBasic = m_rnd.RndInt64Range(iFinancialOutBasicMin, iFinancialOutBasicMax);
|
|
iOutDilut = 0;
|
|
fInvent = m_rnd.RndDoubleIncrRange(fFinancialInventMin, fFinancialInventMax, 0.01);
|
|
fAssets = m_rnd.RndDoubleIncrRange(fFinancialAssetsMin, fFinancialAssetsMax, 0.01);
|
|
fLiab = m_rnd.RndDoubleIncrRange(fFinancialLiabMin, fFinancialLiabMax, 0.01);
|
|
fBasicEPS = 0.00;
|
|
fDilutEPS = 0.00;
|
|
fMargin = 0.00;
|
|
|
|
for (i = 0; i < iFinsPerCompany; ++i) {
|
|
// Compute values for this quarter
|
|
fRev = fRev * m_rnd.RndDoubleIncrRange(fFinDataDownMult, fFinDataUpMult, fFinDataIncr);
|
|
fEarn = fEarn * m_rnd.RndDoubleIncrRange(fFinDataDownMult, fFinDataUpMult, fFinDataIncr);
|
|
if (fEarn >= fRev) { // earnings cannot be greater than the revenue
|
|
fEarn = fEarn * fFinDataDownMult;
|
|
}
|
|
iOutBasic =
|
|
(INT64)((double)iOutBasic * m_rnd.RndDoubleIncrRange(fFinDataDownMult, fFinDataUpMult, fFinDataIncr));
|
|
iOutDilut = (INT64)((double)iOutBasic * fDilutedSharesMultiplier);
|
|
fInvent = fInvent * m_rnd.RndDoubleIncrRange(fFinDataDownMult, fFinDataUpMult, fFinDataIncr);
|
|
fAssets = fAssets * m_rnd.RndDoubleIncrRange(fFinDataDownMult, fFinDataUpMult, fFinDataIncr);
|
|
fLiab = fLiab * m_rnd.RndDoubleIncrRange(fFinDataDownMult, fFinDataUpMult, fFinDataIncr);
|
|
fBasicEPS = fEarn / (double)iOutBasic;
|
|
fDilutEPS = fEarn / (double)iOutDilut;
|
|
fMargin = fEarn / fRev.DollarAmount();
|
|
|
|
// Assign values for this quarter
|
|
m_row.m_financials[i].FI_CO_ID = FI_CO_ID;
|
|
m_row.m_financials[i].FI_YEAR = iFinYear;
|
|
m_row.m_financials[i].FI_QTR = iFinQuarter + 1;
|
|
m_row.m_financials[i].FI_QTR_START_DATE.Set(iFinYear, iFinQuarter * 3 + 1, 1);
|
|
m_row.m_financials[i].FI_REVENUE = fRev.DollarAmount();
|
|
m_row.m_financials[i].FI_NET_EARN = fEarn.DollarAmount();
|
|
m_row.m_financials[i].FI_OUT_BASIC = iOutBasic;
|
|
m_row.m_financials[i].FI_OUT_DILUT = iOutDilut;
|
|
m_row.m_financials[i].FI_INVENTORY = fInvent.DollarAmount();
|
|
m_row.m_financials[i].FI_ASSETS = fAssets.DollarAmount();
|
|
m_row.m_financials[i].FI_LIABILITY = fLiab.DollarAmount();
|
|
m_row.m_financials[i].FI_BASIC_EPS = fBasicEPS.DollarAmount();
|
|
m_row.m_financials[i].FI_DILUT_EPS = fDilutEPS.DollarAmount();
|
|
m_row.m_financials[i].FI_MARGIN = fMargin.DollarAmount();
|
|
|
|
// Increment quarter
|
|
iFinQuarter++;
|
|
if (iFinQuarter == iQuartersInYear) { // reached the last quarter in the year
|
|
iFinQuarter = 0; // start from the first quarter
|
|
++iFinYear; // increment year
|
|
}
|
|
}
|
|
|
|
return m_CompanyTable.GenerateNextCO_ID();
|
|
}
|
|
|
|
/*
|
|
* Reset the state for the next load unit.
|
|
*
|
|
* PARAMETERS:
|
|
* none.
|
|
*
|
|
* RETURNS:
|
|
* none.
|
|
*/
|
|
void InitNextLoadUnit() {
|
|
m_rnd.SetSeed(m_rnd.RndNthElement(RNGSeedTableDefault, (RNGSEED)m_iLastRowNumber * iRNGSkipOneRowFinancial));
|
|
|
|
ClearRecord(); // this is needed for EGenTest to work
|
|
}
|
|
|
|
public:
|
|
CFinancialTable(const DataFileManager &dfm, TIdent iCustomerCount, TIdent iStartFromCustomer)
|
|
: TableTemplate<FINANCIAL_GEN_ROW>(), m_CompanyTable(dfm, iCustomerCount, iStartFromCustomer),
|
|
m_iRowsGeneratedPerCompany(iFinsPerCompany), m_bMoreCompanies(true) {
|
|
// Start year to generate financials.
|
|
// Count by quaters
|
|
m_iFinYear = iDailyMarketBaseYear; // first financial year
|
|
m_iFinQuarter = iDailyMarketBaseMonth / 3; // first financial quarter in the year (0-based)
|
|
|
|
m_bMoreRecords = true; // initialize once
|
|
|
|
m_iFinancialCountForOneLoadUnit =
|
|
dfm.CompanyFile().CalculateCompanyCount(iDefaultLoadUnitSize) * iFinsPerCompany;
|
|
|
|
m_iLastRowNumber = dfm.CompanyFile().CalculateStartFromCompany(iStartFromCustomer) * iFinsPerCompany;
|
|
};
|
|
|
|
bool GenerateNextRecord() {
|
|
// Reset RNG at Load Unit boundary, so that all data is repeatable.
|
|
//
|
|
if (m_iLastRowNumber % m_iFinancialCountForOneLoadUnit == 0) {
|
|
InitNextLoadUnit();
|
|
}
|
|
|
|
++m_iLastRowNumber;
|
|
|
|
++m_iRowsGeneratedPerCompany;
|
|
|
|
if (m_iRowsGeneratedPerCompany >= iFinsPerCompany) {
|
|
if (m_bMoreCompanies) {
|
|
// All rows for the current company have been returned
|
|
// therefore move on to the next company
|
|
m_bMoreCompanies = GenerateFinancialRows();
|
|
|
|
m_iRowsGeneratedPerCompany = 0;
|
|
}
|
|
}
|
|
|
|
// Return false when generated the last row of the last company
|
|
if (!m_bMoreCompanies && (m_iRowsGeneratedPerCompany == iFinsPerCompany - 1)) {
|
|
m_bMoreRecords = false;
|
|
}
|
|
|
|
return MoreRecords();
|
|
}
|
|
|
|
const FINANCIAL_ROW &GetRow() {
|
|
return m_row.m_financials[m_iRowsGeneratedPerCompany];
|
|
}
|
|
};
|
|
|
|
} // namespace TPCE
|
|
|
|
#endif // FINANCIAL_TABLE_H
|