1 | /* |
2 | * Legal Notice |
3 | * |
4 | * This document and associated source code (the "Work") is a part of a |
5 | * benchmark specification maintained by the TPC. |
6 | * |
7 | * The TPC reserves all right, title, and interest to the Work as provided |
8 | * under U.S. and international laws, including without limitation all patent |
9 | * and trademark rights therein. |
10 | * |
11 | * No Warranty |
12 | * |
13 | * 1.1 TO THE MAXIMUM EXTENT PERMITTED BY APPLICABLE LAW, THE INFORMATION |
14 | * CONTAINED HEREIN IS PROVIDED "AS IS" AND WITH ALL FAULTS, AND THE |
15 | * AUTHORS AND DEVELOPERS OF THE WORK HEREBY DISCLAIM ALL OTHER |
16 | * WARRANTIES AND CONDITIONS, EITHER EXPRESS, IMPLIED OR STATUTORY, |
17 | * INCLUDING, BUT NOT LIMITED TO, ANY (IF ANY) IMPLIED WARRANTIES, |
18 | * DUTIES OR CONDITIONS OF MERCHANTABILITY, OF FITNESS FOR A PARTICULAR |
19 | * PURPOSE, OF ACCURACY OR COMPLETENESS OF RESPONSES, OF RESULTS, OF |
20 | * WORKMANLIKE EFFORT, OF LACK OF VIRUSES, AND OF LACK OF NEGLIGENCE. |
21 | * ALSO, THERE IS NO WARRANTY OR CONDITION OF TITLE, QUIET ENJOYMENT, |
22 | * QUIET POSSESSION, CORRESPONDENCE TO DESCRIPTION OR NON-INFRINGEMENT |
23 | * WITH REGARD TO THE WORK. |
24 | * 1.2 IN NO EVENT WILL ANY AUTHOR OR DEVELOPER OF THE WORK BE LIABLE TO |
25 | * ANY OTHER PARTY FOR ANY DAMAGES, INCLUDING BUT NOT LIMITED TO THE |
26 | * COST OF PROCURING SUBSTITUTE GOODS OR SERVICES, LOST PROFITS, LOSS |
27 | * OF USE, LOSS OF DATA, OR ANY INCIDENTAL, CONSEQUENTIAL, DIRECT, |
28 | * INDIRECT, OR SPECIAL DAMAGES WHETHER UNDER CONTRACT, TORT, WARRANTY, |
29 | * OR OTHERWISE, ARISING IN ANY WAY OUT OF THIS OR ANY OTHER AGREEMENT |
30 | * RELATING TO THE WORK, WHETHER OR NOT SUCH AUTHOR OR DEVELOPER HAD |
31 | * ADVANCE NOTICE OF THE POSSIBILITY OF SUCH DAMAGES. |
32 | * |
33 | * Contributors |
34 | * - Sergey Vasilevskiy |
35 | * - Doug Johnson |
36 | */ |
37 | |
38 | #include "main/EGenTables_stdafx.h" |
39 | |
40 | using namespace TPCE; |
41 | |
42 | // Percentages used in determining gender. |
43 | const int iPercentGenderIsMale = 49; |
44 | |
45 | /* |
46 | * Initializes in-memory representation of names files. |
47 | */ |
48 | CPerson::CPerson(const DataFileManager &dfm, TIdent iStartFromCustomer, bool bCacheEnabled) |
49 | : m_LastNames(dfm.LastNameDataFile()), m_MaleFirstNames(dfm.MaleFirstNameDataFile()), |
50 | m_FemaleFirstNames(dfm.FemaleFirstNameDataFile()), m_bCacheEnabled(bCacheEnabled), m_iCacheSize(0), |
51 | m_iCacheOffset(0), INVALID_NAME_CACHE_ENTRY(-1), INVALID_GENDER_CACHE_ENTRY('X') { |
52 | if (m_bCacheEnabled) { |
53 | m_iCacheSize = iDefaultLoadUnitSize; |
54 | m_iCacheOffset = iTIdentShift + iStartFromCustomer; |
55 | m_FirstNameCache = new int[m_iCacheSize]; |
56 | m_LastNameCache = new int[m_iCacheSize]; |
57 | m_GenderCache = new char[m_iCacheSize]; |
58 | for (int i = 0; i < m_iCacheSize; i++) { |
59 | m_FirstNameCache[i] = INVALID_NAME_CACHE_ENTRY; |
60 | m_LastNameCache[i] = INVALID_NAME_CACHE_ENTRY; |
61 | m_GenderCache[i] = INVALID_GENDER_CACHE_ENTRY; |
62 | } |
63 | } |
64 | } |
65 | |
66 | /* |
67 | * Deallocate in-memory representation of names files. |
68 | */ |
69 | CPerson::~CPerson() { |
70 | if (m_bCacheEnabled) { |
71 | delete[] m_FirstNameCache; |
72 | delete[] m_LastNameCache; |
73 | delete[] m_GenderCache; |
74 | } |
75 | } |
76 | |
77 | /* |
78 | * Resets the cache. |
79 | */ |
80 | void CPerson::InitNextLoadUnit(TIdent iCacheOffsetIncrement) { |
81 | if (m_bCacheEnabled) { |
82 | m_iCacheOffset += iCacheOffsetIncrement; |
83 | for (int i = 0; i < m_iCacheSize; i++) { |
84 | m_FirstNameCache[i] = INVALID_NAME_CACHE_ENTRY; |
85 | m_LastNameCache[i] = INVALID_NAME_CACHE_ENTRY; |
86 | m_GenderCache[i] = INVALID_GENDER_CACHE_ENTRY; |
87 | } |
88 | } |
89 | } |
90 | |
91 | /* |
92 | * Returns the last name for a particular customer id. |
93 | * It'll always be the same for the same customer id. |
94 | */ |
95 | const string &CPerson::GetLastName(TIdent CID) { |
96 | return getName<LastNameDataFile_t>(CID, m_LastNames, m_LastNameCache, RNGSeedBaseLastName); |
97 | } |
98 | |
99 | /* |
100 | * Returns the first name for a particular customer id. |
101 | * Determines gender first. |
102 | */ |
103 | const string &CPerson::GetFirstName(TIdent CID) { |
104 | if (IsMaleGender(CID)) { |
105 | return getName<MaleFirstNameDataFile_t>(CID, m_MaleFirstNames, m_FirstNameCache, RNGSeedBaseFirstName); |
106 | } else { |
107 | return getName<FemaleFirstNameDataFile_t>(CID, m_FemaleFirstNames, m_FirstNameCache, RNGSeedBaseFirstName); |
108 | } |
109 | } |
110 | /* |
111 | * Returns the middle name. |
112 | */ |
113 | char CPerson::GetMiddleName(TIdent CID) { |
114 | RNGSEED OldSeed; |
115 | char cMiddleInitial[2]; |
116 | |
117 | OldSeed = m_rnd.GetSeed(); |
118 | m_rnd.SetSeed(m_rnd.RndNthElement(RNGSeedBaseMiddleInitial, (RNGSEED)CID)); |
119 | cMiddleInitial[1] = '\0'; |
120 | m_rnd.RndAlphaNumFormatted(cMiddleInitial, "a" ); |
121 | m_rnd.SetSeed(OldSeed); |
122 | return (cMiddleInitial[0]); |
123 | } |
124 | |
125 | /* |
126 | * Returns the gender character for a particular customer id. |
127 | */ |
128 | char CPerson::GetGender(TIdent CID) { |
129 | RNGSEED OldSeed; |
130 | char cGender; |
131 | |
132 | // It is possible (and expected) to get CID values that are oustide the |
133 | // current load unit. For example, AccountPermission CIDs and Broker IDs |
134 | // can be outside the current load unit. These "out of bounds" CIDs are not |
135 | // cached so we need to account for this. |
136 | TIdent index = CID - m_iCacheOffset; |
137 | bool bCheckCache = (index >= 0 && index < m_iCacheSize); |
138 | |
139 | // Use the cache if we can. |
140 | if (m_bCacheEnabled && bCheckCache && (INVALID_GENDER_CACHE_ENTRY != m_GenderCache[index])) { |
141 | return m_GenderCache[index]; |
142 | } |
143 | |
144 | // We couldn't use the cache. |
145 | |
146 | OldSeed = m_rnd.GetSeed(); |
147 | m_rnd.SetSeed(m_rnd.RndNthElement(RNGSeedBaseGender, (RNGSEED)CID)); |
148 | |
149 | // Find out gender |
150 | if (m_rnd.RndPercent(iPercentGenderIsMale)) { |
151 | cGender = 'M'; |
152 | } else { |
153 | cGender = 'F'; |
154 | } |
155 | |
156 | // Cache the result if appropriate. |
157 | if (m_bCacheEnabled && bCheckCache) { |
158 | m_GenderCache[index] = cGender; |
159 | } |
160 | |
161 | // Restore the RNG |
162 | m_rnd.SetSeed(OldSeed); |
163 | return (cGender); |
164 | } |
165 | |
166 | /* |
167 | * Returns TRUE is a customer id is male |
168 | */ |
169 | bool CPerson::IsMaleGender(TIdent CID) { |
170 | return GetGender(CID) == 'M'; |
171 | } |
172 | |
173 | /* |
174 | * Generate tax id |
175 | */ |
176 | void CPerson::GetTaxID(TIdent CID, char *buf) { |
177 | RNGSEED OldSeed; |
178 | |
179 | OldSeed = m_rnd.GetSeed(); |
180 | |
181 | // NOTE: the call to RndAlphaNumFormatted "consumes" an RNG value |
182 | // for EACH character in the format string. Therefore, to avoid getting |
183 | // tax ID's that overlap N-1 out of N characters, multiply the offset into |
184 | // the sequence by N to get a unique range of values. |
185 | m_rnd.SetSeed(m_rnd.RndNthElement(RNGSeedBaseTaxID, ((RNGSEED)CID * TaxIDFmt_len))); |
186 | m_rnd.RndAlphaNumFormatted(buf, TaxIDFmt); |
187 | m_rnd.SetSeed(OldSeed); |
188 | } |
189 | |
190 | /* |
191 | * Get first name, last name, and tax id. |
192 | */ |
193 | void CPerson::GetFirstLastAndTaxID(TIdent C_ID, char *szFirstName, char *szLastName, char *szTaxID) { |
194 | // Fill in the last name |
195 | strncpy(szLastName, GetLastName(C_ID).c_str(), cL_NAME_len); |
196 | // Fill in the first name |
197 | strncpy(szFirstName, GetFirstName(C_ID).c_str(), cF_NAME_len); |
198 | // Fill in the tax id |
199 | GetTaxID(C_ID, szTaxID); |
200 | } |
201 | |