verilator/src/V3EmitMk.cpp
2024-10-31 14:38:53 -04:00

940 lines
44 KiB
C++

// -*- mode: C++; c-file-style: "cc-mode" -*-
//*************************************************************************
// DESCRIPTION: Verilator: Emit Makefile
//
// Code available from: https://verilator.org
//
//*************************************************************************
//
// Copyright 2004-2024 by Wilson Snyder. This program is free software; you
// can redistribute it and/or modify it under the terms of either the GNU
// Lesser General Public License Version 3 or the Perl Artistic License
// Version 2.0.
// SPDX-License-Identifier: LGPL-3.0-only OR Artistic-2.0
//
//*************************************************************************
#include "V3PchAstNoMT.h" // VL_MT_DISABLED_CODE_UNIT
#include "V3EmitMk.h"
#include "V3EmitCBase.h"
#include "V3HierBlock.h"
#include "V3Os.h"
VL_DEFINE_DEBUG_FUNCTIONS;
// Groups adjacent files in a list, evenly distributing sum of scores
class EmitGroup final {
public:
struct FileOrConcatenatedFilesList final {
const std::string m_filename; // Filename or output group filename if grouping
std::vector<std::string> m_concatenatedFilenames; // Grouped filenames if grouping
bool isConcatenatingFile() const { return !m_concatenatedFilenames.empty(); }
};
struct FilenameWithScore final {
const std::string m_filename; // Input filename
const uint64_t m_score; // Input file complexity
};
private:
// Data of a single work unit used in `singleConcatenatedFilesList()`.
struct WorkList final {
uint64_t m_totalScore = 0; // Sum of scores of included files
std::vector<FilenameWithScore> m_files; // Included filenames
int m_bucketsNum
= 0; // Number of buckets assigned for this list. Used only in concatenable lists.
bool m_isConcatenable = true; // Indicated whether files on this list can be concatenated.
const int m_dbgId; // Work list ID for debugging.
WorkList() = delete;
WorkList(int id)
: m_dbgId{id} {}
};
// MAIN PARAMETERS
// Minimum number of input files required to perform concatenation.
// Concatenation of a small number of files does not give any performance advantages.
// The value has been chosen arbitrarily, most likely could be larger.
static constexpr size_t MIN_FILES_COUNT = 16;
// Concatenation of only a few files most likely does not increase performance.
// The value has been chosen arbitrarily.
static constexpr size_t MIN_FILES_PER_BUCKET = 4;
// MEMBERS
const std::vector<FilenameWithScore>
m_inputFiles; // List of filenames from initial AstCFile list
std::vector<FileOrConcatenatedFilesList>
m_outputFiles; // Output list of files and group files
const uint64_t m_totalScore; // Sum of file scores
const std::string m_groupFilePrefix; // Prefix for output group filenames
std::vector<WorkList> m_workLists; // Lists of small enough files
std::unique_ptr<std::ofstream> m_logp; // Dump file
std::vector<WorkList*> m_concatenableListsByDescSize; // Lists sorted by size, descending
EmitGroup(std::vector<FilenameWithScore> inputFiles, uint64_t totalScore,
std::string groupFilePrefix)
: m_inputFiles{std::move(inputFiles)}
, m_totalScore{totalScore}
, m_groupFilePrefix{groupFilePrefix} {}
// Debug logging: prints scores histogram
void dumpLogScoreHistogram(std::ostream& os) {
constexpr int MAX_BAR_LENGTH = 80;
constexpr int MAX_INTERVALS_NUM = 60;
// All scores arranged in ascending order.
std::vector<uint64_t> sortedScores;
sortedScores.reserve(m_inputFiles.size());
std::transform(m_inputFiles.begin(), m_inputFiles.end(), std::back_inserter(sortedScores),
[](const FilenameWithScore& inputFile) { return inputFile.m_score; });
std::sort(sortedScores.begin(), sortedScores.end());
const int64_t topScore = sortedScores.back();
os << "Top score: " << topScore << endl;
const int maxScoreWidth = std::to_string(topScore).length();
const int64_t intervalsNum = std::min<int64_t>(topScore + 1, MAX_INTERVALS_NUM);
struct Interval final {
uint64_t m_lowerBound = 0;
int m_size = 0;
};
std::vector<Interval> intervals;
intervals.resize(intervalsNum);
intervals[0].m_lowerBound = 0;
for (int i = 1; i < intervalsNum; i++) {
intervals[i].m_lowerBound = (topScore + 1) * i / intervalsNum + 1;
}
for (const uint64_t score : sortedScores) {
const unsigned int ivIdx = score * intervalsNum / (topScore + 1);
++intervals[ivIdx].m_size;
}
int topIntervalSize = 0;
for (const Interval& iv : intervals)
topIntervalSize = std::max(topIntervalSize, iv.m_size);
os << "Input files' scores histogram:" << endl;
for (const Interval& iv : intervals) {
const int scaledSize = iv.m_size * (MAX_BAR_LENGTH + 1) / topIntervalSize;
std::string line = " |" + std::string(scaledSize, '#');
os << std::setw(maxScoreWidth) << iv.m_lowerBound << line << " " << iv.m_size << '\n';
}
os << std::setw(maxScoreWidth) << (topScore + 1) << '\n';
os << endl;
}
// PRIVATE METHODS
// Debug logging: dumps Work Lists and their lists of files
void dumpWorkLists(std::ostream& os) {
os << "Initial Work Lists with their concatenation eligibility status:" << endl;
for (const WorkList& list : m_workLists) {
os << "+ [" << (list.m_isConcatenable ? 'x' : ' ') << "] Work List #" << list.m_dbgId
<< " (num of files: " << list.m_files.size()
<< "; total score: " << list.m_totalScore << ")\n";
if (debug() >= 6 || list.m_files.size() < 4) {
// Log all files
for (const FilenameWithScore& file : list.m_files)
os << "| + " << file.m_filename << " (score: " << file.m_score << ")\n";
} else {
// Log only first and last file
const FilenameWithScore& first = list.m_files.front();
const FilenameWithScore& last = list.m_files.back();
os << "| + " << first.m_filename << " (score: " << first.m_score << ")\n";
os << "| | (... " << (list.m_files.size() - 2) << " files ...)\n";
os << "| + " << last.m_filename << " (score: " << last.m_score << ")\n";
}
}
os << endl;
}
// Debug logging: dumps list of output files. List of grouped files is additionally printed
// for each concatenating file.
void dumpOutputList(std::ostream& os) const {
os << "List of output files after execution of concatenation:" << endl;
for (const FileOrConcatenatedFilesList& entry : m_outputFiles) {
if (entry.isConcatenatingFile()) {
os << "+ " << entry.m_filename << " (concatenating file)\n";
for (const string& f : entry.m_concatenatedFilenames) {
os << "| + " << f << '\n';
}
} else {
os << "+ " << entry.m_filename << '\n';
}
}
}
// Called when the concatenation is aborted, creates an identity mapping
bool fallbackNoGrouping(size_t inputFilesCount) {
const int totalBucketsNum = v3Global.opt.outputGroups();
// Return early if there's nothing to do.
bool groupingRedundant = false;
if (inputFilesCount < MIN_FILES_COUNT) {
UINFO(4, "File concatenation skipped: Too few files ("
<< m_inputFiles.size() << " < " << MIN_FILES_COUNT << ")" << endl);
groupingRedundant = true;
}
if (inputFilesCount < (MIN_FILES_PER_BUCKET * totalBucketsNum)) {
UINFO(4, "File concatenation skipped: Too few files per bucket ("
<< m_inputFiles.size() << " < " << MIN_FILES_PER_BUCKET << " - "
<< totalBucketsNum << ")" << endl);
groupingRedundant = true;
}
if (!groupingRedundant) return false;
m_outputFiles.reserve(m_inputFiles.size());
for (const FilenameWithScore& filename : m_inputFiles) {
m_outputFiles.push_back({std::move(filename.m_filename), {}});
}
return true;
}
void createWorkLists() {
// Create initial Work Lists.
const int totalBucketsNum = v3Global.opt.outputGroups();
// Input files with a score exceeding this value are excluded from concatenation.
const uint64_t concatenableFileMaxScore = m_totalScore / totalBucketsNum / 2;
V3Stats::addStat("Concatenation max score", concatenableFileMaxScore);
int nextWorkListId = 0;
if (m_logp) *m_logp << "Input files with their concatenation eligibility status:" << endl;
for (const FilenameWithScore& inputFile : m_inputFiles) {
const bool fileIsConcatenable = (inputFile.m_score <= concatenableFileMaxScore);
if (m_logp)
*m_logp << " + [" << (fileIsConcatenable ? 'x' : ' ') << "] "
<< inputFile.m_filename << " (score: " << inputFile.m_score << ")"
<< endl;
V3Stats::addStatSum(fileIsConcatenable ? "Concatenation total grouped score"
: "Concatenation total non-grouped score",
inputFile.m_score);
// Add new list if the last list's concatenability does not match the inputFile's
// concatenability
if (m_workLists.empty() || m_workLists.back().m_isConcatenable != fileIsConcatenable) {
m_workLists.push_back(WorkList{nextWorkListId++});
m_workLists.back().m_isConcatenable = fileIsConcatenable;
}
// Add inputFile to the last list
WorkList& list = m_workLists.back();
list.m_files.push_back({inputFile.m_filename, inputFile.m_score});
list.m_totalScore += inputFile.m_score;
}
if (m_logp) *m_logp << endl;
}
void assignBuckets(uint64_t concatenableFilesTotalScore) {
// Assign buckets to lists
const size_t totalBucketsNum = v3Global.opt.outputGroups();
// More concatenable lists than buckets. Exclude lists with lowest number of files.
// Does not happen very often due to files being already filtered out by comparison of
// their score to ConcatenableFilesMaxScore.
if (m_concatenableListsByDescSize.size() > totalBucketsNum) {
// Debugging: Log which work lists will be kept
if (m_logp) {
*m_logp << "More Work Lists than buckets; "
"Work Lists with statuses indicating whether the list will be kept:"
<< endl;
// Only lists that will be kept. List that will be removed are logged below.
std::for_each(m_concatenableListsByDescSize.begin(),
m_concatenableListsByDescSize.begin() + totalBucketsNum,
[&](WorkList* listp) {
*m_logp << "+ [x] Work List #" << listp->m_dbgId
<< " (num of files: " << listp->m_files.size()
<< "; total score: " << listp->m_totalScore << ")\n";
});
}
// NOTE: Not just debug logging - notice `isConcatenable` assignment in the loop.
std::for_each(m_concatenableListsByDescSize.begin() + totalBucketsNum,
m_concatenableListsByDescSize.end(), [&](WorkList* listp) {
listp->m_isConcatenable = false;
if (m_logp)
*m_logp << "+ [ ] Work List #" << listp->m_dbgId
<< " (num of files: " << listp->m_files.size()
<< "; total score: " << listp->m_totalScore << ")\n";
});
if (m_logp) *m_logp << endl;
m_concatenableListsByDescSize.resize(totalBucketsNum);
// Recalculate stats
concatenableFilesTotalScore = 0;
for (WorkList* listp : m_concatenableListsByDescSize) {
concatenableFilesTotalScore += listp->m_totalScore;
}
}
const uint64_t idealBucketScore = concatenableFilesTotalScore / totalBucketsNum;
V3Stats::addStat("Concatenation ideal bucket score", idealBucketScore);
if (m_logp) *m_logp << "Buckets assigned to Work Lists:" << endl;
int availableBuckets = v3Global.opt.outputGroups();
for (WorkList* listp : m_concatenableListsByDescSize) {
if (availableBuckets > 0) {
listp->m_bucketsNum = std::min(
availableBuckets, std::max<int>(1, listp->m_totalScore / idealBucketScore));
availableBuckets -= listp->m_bucketsNum;
if (m_logp)
*m_logp << "+ [" << std::setw(2) << listp->m_bucketsNum << "] Work List #"
<< listp->m_dbgId << '\n';
} else {
// Out of buckets. Instead of recalculating everything just exclude the list.
listp->m_isConcatenable = false;
if (m_logp)
*m_logp << "+ [ 0] Work List #" << std::left << std::setw(4) << listp->m_dbgId
<< std::right << " (excluding from concatenation)\n";
}
}
if (m_logp) *m_logp << endl;
}
void buildOutputList() {
// Assign files to buckets and build final list of files
// At this point the workLists contains concatenatable file lists separated by one or more
// non-concatenable file lists. Each concatenatable list has N buckets (where N > 0)
// assigned to it, which have to be filled with files from this list. Ideally, sum of file
// scores in every bucket should be the same.
int concatenatedFileId = 0;
for (WorkList& list : m_workLists) {
if (!list.m_isConcatenable) {
for (FilenameWithScore& file : list.m_files) {
m_outputFiles.push_back({std::move(file.m_filename), {}});
}
continue;
}
// Ideal bucket score limited to buckets and score of the current Work List.
const uint64_t listIdealBucketScore = list.m_totalScore / list.m_bucketsNum;
auto fileIt = list.m_files.begin();
for (int i = 0; i < list.m_bucketsNum; ++i) {
FileOrConcatenatedFilesList bucket{v3Global.opt.prefix() + "_" + m_groupFilePrefix
+ std::to_string(concatenatedFileId++),
{}};
uint64_t bucketScore = 0;
for (; fileIt != list.m_files.end(); ++fileIt) {
const uint64_t diffNow
= std::abs((int64_t)(listIdealBucketScore - bucketScore));
const uint64_t diffIfAdded = std::abs(
(int64_t)(listIdealBucketScore - bucketScore - fileIt->m_score));
if (bucketScore == 0 || fileIt->m_score == 0 || diffNow > diffIfAdded) {
// Bucket score will be better with the file in it.
bucketScore += fileIt->m_score;
bucket.m_concatenatedFilenames.push_back(std::move(fileIt->m_filename));
} else {
// Best possible bucket score reached, process next bucket.
break;
}
}
const bool lastBucketAndLeftovers
= (i + 1 == list.m_bucketsNum) && (fileIt != list.m_files.end());
if (bucket.m_concatenatedFilenames.size() > 1 || lastBucketAndLeftovers) {
m_outputFiles.push_back(std::move(bucket));
} else if (bucket.m_concatenatedFilenames.size() == 1) {
// Unwrap the bucket if it contains only one file.
m_outputFiles.push_back(
{std::move(bucket.m_concatenatedFilenames.front()), {}});
}
// Most likely no bucket will be empty in normal situations. If it happen the
// bucket will just be dropped.
}
for (; fileIt != list.m_files.end(); ++fileIt) {
// The Work List is out of buckets, but some files were left.
// Add them to the last bucket.
UASSERT(m_outputFiles.back().isConcatenatingFile(),
"Cannot add leftover files to a single file");
m_outputFiles.back().m_concatenatedFilenames.push_back(fileIt->m_filename);
}
}
}
void assertFilesSame() const {
auto ifIt = m_inputFiles.begin();
auto ofIt = m_outputFiles.begin();
while (ifIt != m_inputFiles.end() && ofIt != m_outputFiles.end()) {
if (ofIt->isConcatenatingFile()) {
for (const string& ocf : ofIt->m_concatenatedFilenames) {
UASSERT(ifIt != m_inputFiles.end(),
"More output files than input files. First extra file: " << ocf);
UASSERT(ifIt->m_filename == ocf,
"Name mismatch: " << ifIt->m_filename << " != " << ocf);
++ifIt;
}
++ofIt;
} else {
UASSERT(ifIt->m_filename == ofIt->m_filename,
"Name mismatch: " << ifIt->m_filename << " != " << ofIt->m_filename);
++ifIt;
++ofIt;
}
}
UASSERT(ifIt == m_inputFiles.end(),
"More input files than input files. First extra file: " << ifIt->m_filename);
UASSERT(ofIt == m_outputFiles.end(),
"More output files than input files. First extra file: " << ofIt->m_filename);
}
void process() {
UINFO(4, __FUNCTION__ << ":" << endl);
UINFO(5, "Number of input files: " << m_inputFiles.size() << endl);
UINFO(5, "Total score: " << m_totalScore << endl);
UINFO(5, "Group file prefix: " << m_groupFilePrefix << endl);
const int totalBucketsNum = v3Global.opt.outputGroups();
UINFO(5, "Number of buckets: " << totalBucketsNum << endl);
UASSERT(totalBucketsNum > 0, "More than 0 buckets required");
if (fallbackNoGrouping(m_inputFiles.size())) return;
if (dumpLevel() >= 6) {
const string filename = v3Global.debugFilename("outputgroup") + ".txt";
m_logp = std::unique_ptr<std::ofstream>{V3File::new_ofstream(filename)};
if (m_logp->fail()) v3fatal("Can't write " << filename);
}
if (m_logp) dumpLogScoreHistogram(*m_logp);
createWorkLists();
// Collect stats and mark lists with only one file as non-concatenable
size_t concatenableFilesCount = 0;
int64_t concatenableFilesTotalScore = 0;
for (WorkList& list : m_workLists) {
if (!list.m_isConcatenable) continue;
// "Concatenation" of a single file is pointless
if (list.m_files.size() == 1) {
list.m_isConcatenable = false;
UINFO(5, "Excluding from concatenation: Work List contains only one file: "
"Work List #"
<< list.m_dbgId << endl);
continue;
}
concatenableFilesCount += list.m_files.size();
concatenableFilesTotalScore += list.m_totalScore;
// This vector is sorted below
m_concatenableListsByDescSize.push_back(&list);
}
if (m_logp) dumpWorkLists(*m_logp);
// Check concatenation conditions again using more precise data
if (fallbackNoGrouping(concatenableFilesCount)) return;
std::sort(m_concatenableListsByDescSize.begin(), m_concatenableListsByDescSize.end(),
[](const WorkList* ap, const WorkList* bp) {
// Sort in descending order by number of files
if (ap->m_files.size() != bp->m_files.size())
return (ap->m_files.size() > bp->m_files.size());
// As a fallback sort in ascending order by totalSize. This makes lists
// with higher score more likely to be excluded.
return bp->m_totalScore > ap->m_totalScore;
});
assignBuckets(concatenableFilesTotalScore);
buildOutputList();
if (m_logp) dumpOutputList(*m_logp);
assertFilesSame();
}
public:
static std::vector<FileOrConcatenatedFilesList>
singleConcatenatedFilesList(std::vector<FilenameWithScore> inputFiles, uint64_t totalScore,
std::string groupFilePrefix) {
EmitGroup group{std::move(inputFiles), totalScore, groupFilePrefix};
group.process();
return group.m_outputFiles;
}
};
// ######################################################################
// Emit statements and expressions
class EmitMk final {
using FileOrConcatenatedFilesList = EmitGroup::FileOrConcatenatedFilesList;
using FilenameWithScore = EmitGroup::FilenameWithScore;
public:
// METHODS
static void emitConcatenatingFile(const FileOrConcatenatedFilesList& entry) {
UASSERT(entry.isConcatenatingFile(), "Passed entry does not represent concatenating file");
V3OutCFile concatenatingFile{v3Global.opt.makeDir() + "/" + entry.m_filename + ".cpp"};
concatenatingFile.putsHeader();
for (const string& file : entry.m_concatenatedFilenames) {
concatenatingFile.puts("#include \"" + file + ".cpp\"\n");
}
}
void putMakeClassEntry(V3OutMkFile& of, const string& name) {
of.puts("\t" + V3Os::filenameNonDirExt(name) + " \\\n");
}
void emitClassMake() {
std::vector<FileOrConcatenatedFilesList> vmClassesSlowList;
std::vector<FileOrConcatenatedFilesList> vmClassesFastList;
if (v3Global.opt.outputGroups() > 0) {
std::vector<FilenameWithScore> slowFiles;
std::vector<FilenameWithScore> fastFiles;
uint64_t slowTotalScore = 0;
uint64_t fastTotalScore = 0;
for (AstNodeFile* nodep = v3Global.rootp()->filesp(); nodep;
nodep = VN_AS(nodep->nextp(), NodeFile)) {
const AstCFile* const cfilep = VN_CAST(nodep, CFile);
if (cfilep && cfilep->source() && cfilep->support() == false) {
std::vector<FilenameWithScore>& files = cfilep->slow() ? slowFiles : fastFiles;
uint64_t& totalScore = cfilep->slow() ? slowTotalScore : fastTotalScore;
totalScore += cfilep->complexityScore();
files.push_back(
{V3Os::filenameNonDirExt(cfilep->name()), cfilep->complexityScore()});
}
}
vmClassesSlowList = EmitGroup::singleConcatenatedFilesList(
std::move(slowFiles), slowTotalScore, "vm_classes_Slow_");
vmClassesFastList = EmitGroup::singleConcatenatedFilesList(
std::move(fastFiles), fastTotalScore, "vm_classes_");
}
// Generate the makefile
V3OutMkFile of{v3Global.opt.makeDir() + "/" + v3Global.opt.prefix() + "_classes.mk"};
of.putsHeader();
of.puts("# DESCR"
"IPTION: Verilator output: Make include file with class lists\n");
of.puts("#\n");
of.puts("# This file lists generated Verilated files, for including "
"in higher level makefiles.\n");
of.puts("# See " + v3Global.opt.prefix() + ".mk" + " for the caller.\n");
of.puts("\n### Switches...\n");
of.puts("# C11 constructs required? 0/1 (always on now)\n");
of.puts("VM_C11 = 1\n");
of.puts("# Timing enabled? 0/1\n");
of.puts("VM_TIMING = ");
of.puts(v3Global.usesTiming() ? "1" : "0");
of.puts("\n");
of.puts("# Coverage output mode? 0/1 (from --coverage)\n");
of.puts("VM_COVERAGE = ");
of.puts(v3Global.opt.coverage() ? "1" : "0");
of.puts("\n");
of.puts("# Parallel builds? 0/1 (from --output-split)\n");
of.puts("VM_PARALLEL_BUILDS = ");
of.puts(v3Global.useParallelBuild() ? "1" : "0");
of.puts("\n");
of.puts("# Tracing output mode? 0/1 (from --trace/--trace-fst)\n");
of.puts("VM_TRACE = ");
of.puts(v3Global.opt.trace() ? "1" : "0");
of.puts("\n");
of.puts("# Tracing output mode in VCD format? 0/1 (from --trace)\n");
of.puts("VM_TRACE_VCD = ");
of.puts(v3Global.opt.trace() && v3Global.opt.traceFormat().vcd() ? "1" : "0");
of.puts("\n");
of.puts("# Tracing output mode in FST format? 0/1 (from --trace-fst)\n");
of.puts("VM_TRACE_FST = ");
of.puts(v3Global.opt.trace() && v3Global.opt.traceFormat().fst() ? "1" : "0");
of.puts("\n");
of.puts("\n### Object file lists...\n");
for (int support = 0; support < 3; ++support) {
for (const bool& slow : {false, true}) {
if (support == 2) {
of.puts("# Global classes, need linked once per executable");
} else if (support) {
of.puts("# Generated support classes");
} else {
of.puts("# Generated module classes");
}
if (slow) {
of.puts(", non-fast-path, compile with low/medium optimization\n");
} else {
of.puts(", fast-path, compile with highest optimization\n");
}
of.puts(support == 2 ? "VM_GLOBAL" : support == 1 ? "VM_SUPPORT" : "VM_CLASSES");
of.puts(slow ? "_SLOW" : "_FAST");
of.puts(" += \\\n");
if (support == 2 && v3Global.opt.hierChild()) {
// Do nothing because VM_GLOBAL is necessary per executable. Top module will
// have them.
} else if (support == 2 && !slow) {
putMakeClassEntry(of, "verilated.cpp");
if (v3Global.dpi()) putMakeClassEntry(of, "verilated_dpi.cpp");
if (v3Global.opt.vpi()) putMakeClassEntry(of, "verilated_vpi.cpp");
if (v3Global.opt.savable()) putMakeClassEntry(of, "verilated_save.cpp");
if (v3Global.opt.coverage()) putMakeClassEntry(of, "verilated_cov.cpp");
if (v3Global.opt.trace()) {
putMakeClassEntry(of, v3Global.opt.traceSourceBase() + "_c.cpp");
}
if (v3Global.usesProbDist()) putMakeClassEntry(of, "verilated_probdist.cpp");
if (v3Global.usesTiming()) putMakeClassEntry(of, "verilated_timing.cpp");
if (v3Global.useRandomizeMethods())
putMakeClassEntry(of, "verilated_random.cpp");
putMakeClassEntry(of, "verilated_threads.cpp");
if (v3Global.opt.usesProfiler()) {
putMakeClassEntry(of, "verilated_profiler.cpp");
}
} else if (support == 2 && slow) {
} else if (support == 0 && v3Global.opt.outputGroups() > 0) {
const std::vector<FileOrConcatenatedFilesList>& list
= slow ? vmClassesSlowList : vmClassesFastList;
for (const FileOrConcatenatedFilesList& entry : list) {
if (entry.isConcatenatingFile()) { emitConcatenatingFile(entry); }
putMakeClassEntry(of, entry.m_filename);
}
} else {
for (AstNodeFile* nodep = v3Global.rootp()->filesp(); nodep;
nodep = VN_AS(nodep->nextp(), NodeFile)) {
const AstCFile* const cfilep = VN_CAST(nodep, CFile);
if (cfilep && cfilep->source() && cfilep->slow() == (slow != 0)
&& cfilep->support() == (support != 0)) {
putMakeClassEntry(of, cfilep->name());
}
}
}
of.puts("\n");
}
}
of.puts("\n");
of.putsHeader();
}
void emitOverallMake() {
// Generate the makefile
V3OutMkFile of{v3Global.opt.makeDir() + "/" + v3Global.opt.prefix() + ".mk"};
of.putsHeader();
of.puts("# DESCR"
"IPTION: Verilator output: "
"Makefile for building Verilated archive or executable\n");
of.puts("#\n");
of.puts("# Execute this makefile from the object directory:\n");
of.puts("# make -f " + v3Global.opt.prefix() + ".mk" + "\n");
of.puts("\n");
if (v3Global.opt.exe()) {
of.puts("default: " + v3Global.opt.exeName() + "\n");
} else if (!v3Global.opt.libCreate().empty()) {
of.puts("default: lib" + v3Global.opt.libCreate() + "\n");
} else {
of.puts("default: lib" + v3Global.opt.prefix() + "\n");
}
of.puts("\n### Constants...\n");
of.puts("# Perl executable (from $PERL, defaults to 'perl' if not set)\n");
of.puts("PERL = " + V3OutFormatter::quoteNameControls(V3Options::getenvPERL()) + "\n");
of.puts("# Python3 executable (from $PYTHON3, defaults to 'python3' if not set)\n");
of.puts("PYTHON3 = " + V3OutFormatter::quoteNameControls(V3Options::getenvPYTHON3())
+ "\n");
of.puts("# Path to Verilator kit (from $VERILATOR_ROOT)\n");
of.puts("VERILATOR_ROOT = "
+ V3OutFormatter::quoteNameControls(V3Options::getenvVERILATOR_ROOT()) + "\n");
of.puts("# SystemC include directory with systemc.h (from $SYSTEMC_INCLUDE)\n");
of.puts("SYSTEMC_INCLUDE ?= "s + V3Options::getenvSYSTEMC_INCLUDE() + "\n");
of.puts("# SystemC library directory with libsystemc.a (from $SYSTEMC_LIBDIR)\n");
of.puts("SYSTEMC_LIBDIR ?= "s + V3Options::getenvSYSTEMC_LIBDIR() + "\n");
// Only check it if we really need the value
if (v3Global.opt.systemC() && !V3Options::systemCFound()) {
v3fatal("Need $SYSTEMC_INCLUDE in environment or when Verilator configured,\n"
"and need $SYSTEMC_LIBDIR in environment or when Verilator configured\n"
"Probably System-C isn't installed, see http://www.systemc.org\n");
}
of.puts("\n### Switches...\n");
of.puts("# C++ code coverage 0/1 (from --prof-c)\n");
of.puts("VM_PROFC = "s + ((v3Global.opt.profC()) ? "1" : "0") + "\n");
of.puts("# SystemC output mode? 0/1 (from --sc)\n");
of.puts("VM_SC = "s + ((v3Global.opt.systemC()) ? "1" : "0") + "\n");
of.puts("# Legacy or SystemC output mode? 0/1 (from --sc)\n");
of.puts("VM_SP_OR_SC = $(VM_SC)\n");
of.puts("# Deprecated\n");
of.puts("VM_PCLI = "s + (v3Global.opt.systemC() ? "0" : "1") + "\n");
of.puts(
"# Deprecated: SystemC architecture to find link library path (from $SYSTEMC_ARCH)\n");
of.puts("VM_SC_TARGET_ARCH = "s + V3Options::getenvSYSTEMC_ARCH() + "\n");
of.puts("\n### Vars...\n");
of.puts("# Design prefix (from --prefix)\n");
of.puts("VM_PREFIX = "s + v3Global.opt.prefix() + "\n");
of.puts("# Module prefix (from --prefix)\n");
of.puts("VM_MODPREFIX = "s + v3Global.opt.modPrefix() + "\n");
of.puts("# User CFLAGS (from -CFLAGS on Verilator command line)\n");
of.puts("VM_USER_CFLAGS = \\\n");
const std::string solver = V3Options::getenvVERILATOR_SOLVER();
if (v3Global.useRandomizeMethods() && solver != "")
of.puts("\t-DVM_SOLVER_DEFAULT='\"" + V3OutFormatter::quoteNameControls(solver)
+ "\"' \\\n");
if (!v3Global.opt.libCreate().empty()) of.puts("\t-fPIC \\\n");
const V3StringList& cFlags = v3Global.opt.cFlags();
for (const string& i : cFlags) of.puts("\t" + i + " \\\n");
of.puts("\n");
of.puts("# User LDLIBS (from -LDFLAGS on Verilator command line)\n");
of.puts("VM_USER_LDLIBS = \\\n");
const V3StringList& ldLibs = v3Global.opt.ldLibs();
for (const string& i : ldLibs) of.puts("\t" + i + " \\\n");
of.puts("\n");
V3StringSet dirs;
of.puts("# User .cpp files (from .cpp's on Verilator command line)\n");
of.puts("VM_USER_CLASSES = \\\n");
const V3StringSet& cppFiles = v3Global.opt.cppFiles();
for (const auto& cppfile : cppFiles) {
of.puts("\t" + V3Os::filenameNonDirExt(cppfile) + " \\\n");
const string dir
= V3Os::filenameRelativePath(V3Os::filenameDir(cppfile), v3Global.opt.makeDir());
dirs.insert(dir);
}
dirs.insert(V3Os::filenameRelativePath(".", v3Global.opt.makeDir()));
of.puts("\n");
of.puts("# User .cpp directories (from .cpp's on Verilator command line)\n");
of.puts("VM_USER_DIR = \\\n");
for (const auto& i : dirs) of.puts("\t" + i + " \\\n");
of.puts("\n");
of.puts("\n### Default rules...\n");
of.puts("# Include list of all generated classes\n");
of.puts("include " + v3Global.opt.prefix() + "_classes.mk\n");
if (v3Global.opt.hierTop()) {
of.puts("# Include rules for hierarchical blocks\n");
of.puts("include " + v3Global.opt.prefix() + "_hier.mk\n");
}
of.puts("# Include global rules\n");
of.puts("include $(VERILATOR_ROOT)/include/verilated.mk\n");
if (v3Global.opt.exe()) {
of.puts("\n### Executable rules... (from --exe)\n");
of.puts("VPATH += $(VM_USER_DIR)\n");
of.puts("\n");
}
const string compilerIncludePch
= v3Global.opt.compilerIncludes().empty() ? "" : "$(VK_PCH_H).fast.gch";
const string compilerIncludeFlag
= v3Global.opt.compilerIncludes().empty() ? "" : "$(VK_PCH_I_FAST)";
for (const string& cppfile : cppFiles) {
const string basename = V3Os::filenameNonDirExt(cppfile);
// NOLINTNEXTLINE(performance-inefficient-string-concatenation)
of.puts(basename + ".o: " + cppfile + " " + compilerIncludePch + "\n");
// NOLINTNEXTLINE(performance-inefficient-string-concatenation)
of.puts("\t$(OBJCACHE) $(CXX) $(CXXFLAGS) $(CPPFLAGS) $(OPT_FAST) "
+ compilerIncludeFlag + " -c -o $@ $<\n");
}
if (v3Global.opt.exe()) {
of.puts("\n### Link rules... (from --exe)\n");
// let default rule depend on '{prefix}__ALL.a', for compatibility
of.puts(v3Global.opt.exeName()
+ ": $(VK_USER_OBJS) $(VK_GLOBAL_OBJS) $(VM_PREFIX)__ALL.a $(VM_HIER_LIBS)\n");
of.puts("\t$(LINK) $(LDFLAGS) $^ $(LOADLIBES) $(LDLIBS) $(LIBS) $(SC_LIBS) -o $@\n");
of.puts("\n");
} else if (!v3Global.opt.libCreate().empty()) {
const string libCreateDeps = "$(VK_OBJS) $(VK_USER_OBJS) $(VK_GLOBAL_OBJS) "
+ v3Global.opt.libCreate() + ".o $(VM_HIER_LIBS)";
of.puts("\n### Library rules from --lib-create\n");
// The rule to create .a is defined in verilated.mk, so just define dependency here.
of.puts(v3Global.opt.libCreateName(false) + ": " + libCreateDeps + "\n");
of.puts("\n");
if (v3Global.opt.hierChild()) {
// Hierarchical child does not need .so because hierTop() will create .so from .a
of.puts("lib" + v3Global.opt.libCreate() + ": " + v3Global.opt.libCreateName(false)
+ "\n");
} else {
of.puts(v3Global.opt.libCreateName(true) + ": " + libCreateDeps + "\n");
// Linker on mac emits an error if all symbols are not found here,
// but some symbols that are referred as "DPI-C" can not be found at this moment.
// So add dynamic_lookup
of.puts("ifeq ($(shell uname -s),Darwin)\n");
of.puts("\t$(OBJCACHE) $(CXX) $(CXXFLAGS) $(CPPFLAGS) $(OPT_FAST) -undefined "
"dynamic_lookup -shared -flat_namespace -o $@ $^\n");
of.puts("else\n");
of.puts(
"\t$(OBJCACHE) $(CXX) $(CXXFLAGS) $(CPPFLAGS) $(OPT_FAST) -shared -o $@ $^\n");
of.puts("endif\n");
of.puts("\n");
of.puts("lib" + v3Global.opt.libCreate() + ": " + v3Global.opt.libCreateName(false)
+ " " + v3Global.opt.libCreateName(true) + "\n");
}
} else {
const string libname = "lib" + v3Global.opt.prefix() + ".a";
of.puts("\n### Library rules (default lib mode)\n");
// The rule to create .a is defined in verilated.mk, so just define dependency here.
of.puts(libname + ": $(VK_OBJS) $(VK_USER_OBJS) $(VM_HIER_LIBS)\n");
of.puts("libverilated.a: $(VK_GLOBAL_OBJS)\n");
// let default rule depend on '{prefix}__ALL.a', for compatibility
of.puts("lib" + v3Global.opt.prefix() + ": " + libname
+ " libverilated.a $(VM_PREFIX)__ALL.a\n");
}
of.puts("\n");
of.putsHeader();
}
explicit EmitMk() {
emitClassMake();
emitOverallMake();
}
virtual ~EmitMk() = default;
};
//######################################################################
class EmitMkHierVerilation final {
const V3HierBlockPlan* const m_planp;
const string m_makefile; // path of this makefile
void emitCommonOpts(V3OutMkFile& of) const {
const string cwd = V3Os::filenameRealPath(".");
of.puts("# Verilation of hierarchical blocks are executed in this directory\n");
of.puts("VM_HIER_RUN_DIR := " + cwd + "\n");
of.puts("# Common options for hierarchical blocks\n");
const string fullpath_bin = V3Os::filenameRealPath(v3Global.opt.buildDepBin());
const string verilator_wrapper = V3Os::filenameDir(fullpath_bin) + "/verilator";
of.puts("VM_HIER_VERILATOR := " + verilator_wrapper + "\n");
of.puts("VM_HIER_INPUT_FILES := \\\n");
const V3StringList& vFiles = v3Global.opt.vFiles();
for (const string& i : vFiles) of.puts("\t" + V3Os::filenameRealPath(i) + " \\\n");
of.puts("\n");
const V3StringSet& libraryFiles = v3Global.opt.libraryFiles();
of.puts("VM_HIER_VERILOG_LIBS := \\\n");
for (const string& i : libraryFiles) {
of.puts("\t" + V3Os::filenameRealPath(i) + " \\\n");
}
of.puts("\n");
}
void emitLaunchVerilator(V3OutMkFile& of, const string& argsFile) const {
of.puts("\t@$(MAKE) -C $(VM_HIER_RUN_DIR) -f " + m_makefile
+ " hier_launch_verilator \\\n");
of.puts("\t\tVM_HIER_LAUNCH_VERILATOR_ARGSFILE=\"" + argsFile + "\"\n");
}
void emit(V3OutMkFile& of) const {
of.puts("# Hierarchical Verilation -*- Makefile -*-\n");
of.puts("# DESCR"
"IPTION: Verilator output: Makefile for hierarchical Verilation\n");
of.puts("#\n");
of.puts("# The main makefile " + v3Global.opt.prefix() + ".mk calls this makefile\n");
of.puts("\n");
of.puts("ifndef VM_HIER_VERILATION_INCLUDED\n");
of.puts("VM_HIER_VERILATION_INCLUDED = 1\n\n");
of.puts(".SUFFIXES:\n");
of.puts(".PHONY: hier_build hier_verilation hier_launch_verilator\n");
of.puts("# Libraries of hierarchical blocks\n");
of.puts("VM_HIER_LIBS := \\\n");
const V3HierBlockPlan::HierVector blocks
= m_planp->hierBlocksSorted(); // leaf comes first
// List in order of leaf-last order so that linker can resolve dependency
for (const auto& block : vlstd::reverse_view(blocks)) {
of.puts("\t" + block->hierLibFilename(true) + " \\\n");
}
of.puts("\n");
// Build hierarchical libraries as soon as possible to get maximum parallelism
of.puts("hier_build: $(VM_HIER_LIBS) " + v3Global.opt.prefix() + ".mk\n");
of.puts("\t$(MAKE) -f " + v3Global.opt.prefix() + ".mk\n");
of.puts("hier_verilation: " + v3Global.opt.prefix() + ".mk\n");
emitCommonOpts(of);
// Instead of direct execute of "cd $(VM_HIER_RUN_DIR) && $(VM_HIER_VERILATOR)",
// call via make to get message of "Entering directory" and "Leaving directory".
// This will make some editors and IDEs happy when viewing a logfile.
of.puts("# VM_HIER_LAUNCH_VERILATOR_ARGSFILE must be passed as a command argument\n");
of.puts("hier_launch_verilator:\n");
of.puts("\t$(VM_HIER_VERILATOR) -f $(VM_HIER_LAUNCH_VERILATOR_ARGSFILE)\n");
// Top level module
{
const string argsFile = v3Global.hierPlanp()->topCommandArgsFilename(false);
of.puts("\n# Verilate the top module\n");
of.puts(v3Global.opt.prefix()
+ ".mk: $(VM_HIER_INPUT_FILES) $(VM_HIER_VERILOG_LIBS) ");
of.puts(V3Os::filenameNonDir(argsFile) + " ");
for (const auto& itr : *m_planp) of.puts(itr.second->hierWrapperFilename(true) + " ");
of.puts("\n");
emitLaunchVerilator(of, argsFile);
}
// Rules to process hierarchical blocks
of.puts("\n# Verilate hierarchical blocks\n");
for (const V3HierBlock* const blockp : m_planp->hierBlocksSorted()) {
const string prefix = blockp->hierPrefix();
const string argsFilename = blockp->commandArgsFilename(false);
of.puts(blockp->hierGeneratedFilenames(true));
of.puts(": $(VM_HIER_INPUT_FILES) $(VM_HIER_VERILOG_LIBS) ");
of.puts(V3Os::filenameNonDir(argsFilename) + " ");
const V3HierBlock::HierBlockSet& children = blockp->children();
for (V3HierBlock::HierBlockSet::const_iterator child = children.begin();
child != children.end(); ++child) {
of.puts((*child)->hierWrapperFilename(true) + " ");
}
of.puts("\n");
emitLaunchVerilator(of, argsFilename);
// Rule to build lib*.a
of.puts(blockp->hierLibFilename(true));
of.puts(": ");
of.puts(blockp->hierMkFilename(true));
of.puts(" ");
for (V3HierBlock::HierBlockSet::const_iterator child = children.begin();
child != children.end(); ++child) {
of.puts((*child)->hierLibFilename(true));
of.puts(" ");
}
of.puts("\n\t$(MAKE) -f " + blockp->hierMkFilename(false) + " -C " + prefix);
of.puts(" VM_PREFIX=" + prefix);
of.puts("\n\n");
}
of.puts("endif # Guard\n");
}
public:
explicit EmitMkHierVerilation(const V3HierBlockPlan* planp)
: m_planp{planp}
, m_makefile{v3Global.opt.makeDir() + "/" + v3Global.opt.prefix() + "_hier.mk"} {
V3OutMkFile of{m_makefile};
emit(of);
}
};
//######################################################################
// Gate class functions
void V3EmitMk::emitmk() {
UINFO(2, __FUNCTION__ << ": " << endl);
const EmitMk emitter;
}
void V3EmitMk::emitHierVerilation(const V3HierBlockPlan* planp) {
UINFO(2, __FUNCTION__ << ": " << endl);
EmitMkHierVerilation{planp};
}