Don't emit empty files with low split limits (#2961)

Attempt to split output files when we know a function will actually be
emitted, otherwise we might end up with empty files.
This commit is contained in:
Geza Lore 2021-05-15 16:05:24 +01:00 committed by GitHub
parent 5e95cc9280
commit 828f78ece4
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23

View File

@ -1343,6 +1343,7 @@ unsigned EmitVarTspSorter::s_serialNext = 0;
class EmitCImp final : EmitCStmts { class EmitCImp final : EmitCStmts {
// MEMBERS // MEMBERS
AstNodeModule* m_modp = nullptr; AstNodeModule* m_modp = nullptr;
AstNodeModule* m_fileModp = nullptr; // Files (names, headers) constructed using this module
std::vector<AstChangeDet*> m_blkChangeDetVec; // All encountered changes in block std::vector<AstChangeDet*> m_blkChangeDetVec; // All encountered changes in block
bool m_slow = false; // Creating __Slow file bool m_slow = false; // Creating __Slow file
bool m_fast = false; // Creating non __Slow file (or both) bool m_fast = false; // Creating non __Slow file (or both)
@ -1391,8 +1392,8 @@ class EmitCImp final : EmitCStmts {
} }
} }
V3OutCFile* newOutCFile(AstNodeModule* modp, bool slow, bool source, int filenum = 0) { V3OutCFile* newOutCFile(bool slow, bool source, int filenum = 0) {
string filenameNoExt = v3Global.opt.makeDir() + "/" + prefixNameProtect(modp); string filenameNoExt = v3Global.opt.makeDir() + "/" + prefixNameProtect(m_fileModp);
if (filenum) filenameNoExt += "__" + cvtToStr(filenum); if (filenum) filenameNoExt += "__" + cvtToStr(filenum);
filenameNoExt += (slow ? "__Slow" : ""); filenameNoExt += (slow ? "__Slow" : "");
V3OutCFile* ofp = nullptr; V3OutCFile* ofp = nullptr;
@ -1413,7 +1414,7 @@ class EmitCImp final : EmitCStmts {
} }
ofp->putsHeader(); ofp->putsHeader();
if (modp->isTop() && !source) { if (m_fileModp->isTop() && !source) {
ofp->puts("// DESCR" ofp->puts("// DESCR"
"IPTION: Verilator output: Primary design header\n"); "IPTION: Verilator output: Primary design header\n");
ofp->puts("//\n"); ofp->puts("//\n");
@ -1501,7 +1502,10 @@ class EmitCImp final : EmitCStmts {
} }
virtual void visit(AstMTaskBody* nodep) override { virtual void visit(AstMTaskBody* nodep) override {
ExecMTask* mtp = nodep->execMTaskp(); maybeSplit();
splitSizeInc(10);
const ExecMTask* const mtp = nodep->execMTaskp();
puts("\n"); puts("\n");
puts("void "); puts("void ");
puts(prefixNameProtect(m_modp) + "::" + protect(mtp->cFuncName())); puts(prefixNameProtect(m_modp) + "::" + protect(mtp->cFuncName()));
@ -1525,6 +1529,8 @@ class EmitCImp final : EmitCStmts {
if (nodep->dpiImport()) return; if (nodep->dpiImport()) return;
if (!(nodep->slow() ? m_slow : m_fast)) return; if (!(nodep->slow() ? m_slow : m_fast)) return;
maybeSplit();
m_blkChangeDetVec.clear(); m_blkChangeDetVec.clear();
splitSizeInc(nodep); splitSizeInc(nodep);
@ -1868,8 +1874,8 @@ class EmitCImp final : EmitCStmts {
void emitSavableImp(AstNodeModule* modp); void emitSavableImp(AstNodeModule* modp);
void emitTextSection(AstType type); void emitTextSection(AstType type);
// High level // High level
void emitImpTop(AstNodeModule* modp); void emitImpTop();
void emitImp(AstNodeModule* fileModp, AstNodeModule* modp); void emitImp(AstNodeModule* modp);
void emitSettleLoop(const std::string& eval_call, bool initial); void emitSettleLoop(const std::string& eval_call, bool initial);
void emitWrapEval(AstNodeModule* modp); void emitWrapEval(AstNodeModule* modp);
void emitWrapFast(AstNodeModule* modp); void emitWrapFast(AstNodeModule* modp);
@ -1877,7 +1883,7 @@ class EmitCImp final : EmitCStmts {
void emitMTaskVertexCtors(bool* firstp); void emitMTaskVertexCtors(bool* firstp);
void emitIntTop(AstNodeModule* modp); void emitIntTop(AstNodeModule* modp);
void emitInt(AstNodeModule* modp); void emitInt(AstNodeModule* modp);
void maybeSplit(AstNodeModule* modp); void maybeSplit();
public: public:
EmitCImp() = default; EmitCImp() = default;
@ -3351,9 +3357,9 @@ void EmitCImp::emitInt(AstNodeModule* modp) {
//---------------------------------------------------------------------- //----------------------------------------------------------------------
void EmitCImp::emitImpTop(AstNodeModule* fileModp) { void EmitCImp::emitImpTop() {
puts("\n"); puts("\n");
puts("#include \"" + prefixNameProtect(fileModp) + ".h\"\n"); puts("#include \"" + prefixNameProtect(m_fileModp) + ".h\"\n");
puts("#include \"" + symClassName() + ".h\"\n"); puts("#include \"" + symClassName() + ".h\"\n");
if (v3Global.dpi()) { if (v3Global.dpi()) {
@ -3361,13 +3367,13 @@ void EmitCImp::emitImpTop(AstNodeModule* fileModp) {
puts("#include \"verilated_dpi.h\"\n"); puts("#include \"verilated_dpi.h\"\n");
} }
emitModCUse(fileModp, VUseType::IMP_INCLUDE); emitModCUse(m_fileModp, VUseType::IMP_INCLUDE);
emitModCUse(fileModp, VUseType::IMP_FWD_CLASS); emitModCUse(m_fileModp, VUseType::IMP_FWD_CLASS);
emitTextSection(AstType::atScImpHdr); emitTextSection(AstType::atScImpHdr);
} }
void EmitCImp::emitImp(AstNodeModule* fileModp, AstNodeModule* modp) { void EmitCImp::emitImp(AstNodeModule* modp) {
puts("\n//==========\n"); puts("\n//==========\n");
if (m_slow) { if (m_slow) {
string section; string section;
@ -3389,37 +3395,33 @@ void EmitCImp::emitImp(AstNodeModule* fileModp, AstNodeModule* modp) {
// Blocks // Blocks
for (AstNode* nodep = modp->stmtsp(); nodep; nodep = nodep->nextp()) { for (AstNode* nodep = modp->stmtsp(); nodep; nodep = nodep->nextp()) {
if (AstCFunc* funcp = VN_CAST(nodep, CFunc)) { if (AstCFunc* funcp = VN_CAST(nodep, CFunc)) { mainDoFunc(funcp); }
maybeSplit(fileModp);
mainDoFunc(funcp);
}
} }
} }
//###################################################################### //######################################################################
void EmitCImp::maybeSplit(AstNodeModule* fileModp) { void EmitCImp::maybeSplit() {
if (splitNeeded()) { if (!splitNeeded()) return;
// Splitting file, so using parallel build.
v3Global.useParallelBuild(true); // Splitting file, so using parallel build.
// Close old file v3Global.useParallelBuild(true);
VL_DO_CLEAR(delete m_ofp, m_ofp = nullptr); // Close old file
// Open a new file VL_DO_CLEAR(delete m_ofp, m_ofp = nullptr);
m_ofp = newOutCFile(fileModp, !m_fast, true /*source*/, splitFilenumInc()); // Open a new file
emitImpTop(fileModp); m_ofp = newOutCFile(!m_fast, true /*source*/, splitFilenumInc());
} emitImpTop();
splitSizeInc(10); // Even blank functions get a file with a low csplit
} }
void EmitCImp::mainInt(AstNodeModule* modp) { void EmitCImp::mainInt(AstNodeModule* modp) {
AstNodeModule* fileModp = modp; // Filename constructed using this module
m_modp = modp; m_modp = modp;
m_fileModp = modp;
m_slow = true; m_slow = true;
m_fast = true; m_fast = true;
UINFO(5, " Emitting " << prefixNameProtect(modp) << endl); UINFO(5, " Emitting " << prefixNameProtect(modp) << endl);
m_ofp = newOutCFile(fileModp, false /*slow*/, false /*source*/); m_ofp = newOutCFile(false /*slow*/, false /*source*/);
emitIntTop(modp); emitIntTop(modp);
emitInt(modp); emitInt(modp);
if (AstClassPackage* packagep = VN_CAST(modp, ClassPackage)) { if (AstClassPackage* packagep = VN_CAST(modp, ClassPackage)) {
@ -3434,23 +3436,23 @@ void EmitCImp::mainInt(AstNodeModule* modp) {
void EmitCImp::mainImp(AstNodeModule* modp, bool slow) { void EmitCImp::mainImp(AstNodeModule* modp, bool slow) {
// Output a module // Output a module
AstNodeModule* fileModp = modp; // Filename constructed using this module
m_modp = modp; m_modp = modp;
m_fileModp = modp;
m_slow = slow; m_slow = slow;
m_fast = !slow; m_fast = !slow;
UINFO(5, " Emitting " << prefixNameProtect(modp) << endl); UINFO(5, " Emitting " << prefixNameProtect(modp) << endl);
m_ofp = newOutCFile(fileModp, !m_fast, true /*source*/); m_ofp = newOutCFile(!m_fast, true /*source*/);
emitImpTop(fileModp); emitImpTop();
emitImp(fileModp, modp); emitImp(modp);
if (AstClassPackage* packagep = VN_CAST(modp, ClassPackage)) { if (AstClassPackage* packagep = VN_CAST(modp, ClassPackage)) {
// Put the non-static class implementation in same C++ files as // Put the non-static class implementation in same C++ files as
// often optimizations are possible when both are seen by the // often optimizations are possible when both are seen by the
// compiler together // compiler together
m_modp = packagep->classp(); m_modp = packagep->classp();
emitImp(fileModp, packagep->classp()); emitImp(packagep->classp());
m_modp = modp; m_modp = modp;
} }
@ -3463,7 +3465,6 @@ void EmitCImp::mainImp(AstNodeModule* modp, bool slow) {
vxp = vxp->verticesNextp()) { vxp = vxp->verticesNextp()) {
const ExecMTask* mtaskp = dynamic_cast<const ExecMTask*>(vxp); const ExecMTask* mtaskp = dynamic_cast<const ExecMTask*>(vxp);
if (mtaskp->threadRoot()) { if (mtaskp->threadRoot()) {
maybeSplit(fileModp);
// Only define one function for all the mtasks packed on // Only define one function for all the mtasks packed on
// a given thread. We'll name this function after the // a given thread. We'll name this function after the
// root mtask though it contains multiple mtasks' worth // root mtask though it contains multiple mtasks' worth
@ -3938,21 +3939,29 @@ void V3EmitC::emitc() {
for (AstNodeModule* nodep = v3Global.rootp()->modulesp(); nodep; for (AstNodeModule* nodep = v3Global.rootp()->modulesp(); nodep;
nodep = VN_CAST(nodep->nextp(), NodeModule)) { nodep = VN_CAST(nodep->nextp(), NodeModule)) {
if (VN_IS(nodep, Class)) continue; // Imped with ClassPackage if (VN_IS(nodep, Class)) continue; // Imped with ClassPackage
// clang-format off {
EmitCImp cint; cint.mainInt(nodep); EmitCImp cint;
cint.mainImp(nodep, true); cint.mainInt(nodep);
{ EmitCImp fast; fast.mainImp(nodep, false); } cint.mainImp(nodep, true);
// clang-format on }
{
EmitCImp fast;
fast.mainImp(nodep, false);
}
} }
} }
void V3EmitC::emitcTrace() { void V3EmitC::emitcTrace() {
UINFO(2, __FUNCTION__ << ": " << endl); UINFO(2, __FUNCTION__ << ": " << endl);
if (v3Global.opt.trace()) { if (v3Global.opt.trace()) {
// clang-format off {
{ EmitCTrace slow(true); slow.main(); } EmitCTrace slow(true);
{ EmitCTrace fast(false); fast.main(); } slow.main();
// clang-format on }
{
EmitCTrace fast(false);
fast.main();
}
} }
} }