summaryrefslogtreecommitdiff
path: root/src/exports.cc
diff options
context:
space:
mode:
authorAdrian Thurston <thurston@colm.net>2020-03-14 15:29:52 +0200
committerAdrian Thurston <thurston@colm.net>2020-03-14 15:29:52 +0200
commitf653735830d537715f2885bd832cf04851d35401 (patch)
tree95e6551e39407543366d4f49aedf7b78c6e8bbe1 /src/exports.cc
parentbcc54d5df10cf425e7134b06f70d7ffe1abee4e4 (diff)
downloadcolm-f653735830d537715f2885bd832cf04851d35401.tar.gz
moved source files into commit repository
Diffstat (limited to 'src/exports.cc')
-rw-r--r--src/exports.cc260
1 files changed, 260 insertions, 0 deletions
diff --git a/src/exports.cc b/src/exports.cc
new file mode 100644
index 00000000..988499db
--- /dev/null
+++ b/src/exports.cc
@@ -0,0 +1,260 @@
+/*
+ * Copyright 2006-2018 Adrian Thurston <thurston@colm.net>
+ *
+ * Permission is hereby granted, free of charge, to any person obtaining a copy
+ * of this software and associated documentation files (the "Software"), to
+ * deal in the Software without restriction, including without limitation the
+ * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
+ * sell copies of the Software, and to permit persons to whom the Software is
+ * furnished to do so, subject to the following conditions:
+ *
+ * The above copyright notice and this permission notice shall be included in all
+ * copies or substantial portions of the Software.
+ *
+ * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+ * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+ * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
+ * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+ * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+ * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
+ * SOFTWARE.
+ */
+
+#include <stdbool.h>
+
+#include <iostream>
+
+#include "fsmcodegen.h"
+
+using std::ostream;
+using std::ostringstream;
+using std::string;
+using std::cerr;
+using std::endl;
+
+void Compiler::openNameSpace( ostream &out, Namespace *nspace )
+{
+ if ( nspace == rootNamespace )
+ return;
+
+ openNameSpace( out, nspace->parentNamespace );
+ out << "namespace " << nspace->name << " { ";
+}
+
+void Compiler::closeNameSpace( ostream &out, Namespace *nspace )
+{
+ if ( nspace == rootNamespace )
+ return;
+
+ openNameSpace( out, nspace->parentNamespace );
+ out << " }";
+}
+
+void Compiler::generateExports()
+{
+ ostream &out = *outStream;
+
+ out <<
+ "#ifndef _EXPORTS_H\n"
+ "#define _EXPORTS_H\n"
+ "\n"
+ "#include <colm/colm.h>\n"
+ "#include <colm/tree.h>\n"
+ "#include <colm/colmex.h>\n"
+ "#include <string>\n"
+ "\n";
+
+ /* Declare. */
+ for ( LelList::Iter lel = langEls; lel.lte(); lel++ ) {
+ if ( lel->isEOF )
+ continue;
+
+ openNameSpace( out, lel->nspace );
+ out << "struct " << lel->fullName << ";";
+ closeNameSpace( out, lel->nspace );
+ out << "\n";
+ }
+
+ /* Class definitions. */
+ for ( LelList::Iter lel = langEls; lel.lte(); lel++ ) {
+ if ( lel->isEOF )
+ continue;
+
+ openNameSpace( out, lel->nspace );
+ out << "struct " << lel->fullName << "\n";
+ out << " : public ExportTree\n";
+ out << "{\n";
+ out << " static const int ID = " << lel->id << ";\n";
+
+ if ( mainReturnUT != 0 && mainReturnUT->langEl == lel ) {
+ out << " " << lel->fullName <<
+ "( colm_program *prg ) : ExportTree( prg, returnVal(prg) ) {\n";
+ out << " }\n";
+ }
+
+ out << " " << lel->fullName <<
+ "( colm_program *prg, colm_tree *tree ) : ExportTree( prg, tree ) {\n";
+
+ out << "}\n";
+
+ if ( lel->objectDef != 0 ) {
+ FieldList &fieldList = lel->objectDef->fieldList;
+ for ( FieldList::Iter ofi = fieldList; ofi.lte(); ofi++ ) {
+ ObjectField *field = ofi->value;
+ if ( ( field->useOffset() && field->typeRef != 0 ) || field->isRhsGet() ) {
+ UniqueType *ut = field->typeRef->resolveType( this );
+
+ if ( ut != 0 && ut->typeId == TYPE_TREE )
+ out << " " << ut->langEl->refName << " " << field->name << "();\n";
+ }
+ }
+ }
+
+ bool prodNames = false;
+ for ( LelDefList::Iter prod = lel->defList; prod.lte(); prod++ ) {
+ if ( prod->_name.length() > 0 )
+ prodNames = true;
+ }
+
+ if ( prodNames ) {
+ out << " enum prod_name {\n";
+ for ( LelDefList::Iter prod = lel->defList; prod.lte(); prod++ ) {
+ if ( prod->_name.length() > 0 )
+ out << "\t\t" << prod->_name << " = " << prod->prodNum << ",\n";
+ }
+ out << " };\n";
+ out << " enum prod_name prodName() " <<
+ "{ return (enum prod_name)__tree->prod_num; }\n";
+ }
+
+ out << "};";
+ closeNameSpace( out, lel->nspace );
+ out << "\n";
+ }
+
+ for ( FieldList::Iter of = globalObjectDef->fieldList; of.lte(); of++ ) {
+ ObjectField *field = of->value;
+ if ( field->isExport ) {
+ UniqueType *ut = field->typeRef->resolveType(this);
+ if ( ut != 0 && ut->typeId == TYPE_TREE ) {
+ out << ut->langEl->refName << " " << field->name << "( colm_program *prg );\n";
+ }
+ }
+ }
+
+ out << "\n";
+
+ for ( FunctionList::Iter func = functionList; func.lte(); func++ ) {
+ if ( func->exprt ) {
+ char *refName = func->typeRef->uniqueType->langEl->refName;
+ int paramCount = func->paramList->length();
+ out <<
+ refName << " " << func->name << "( colm_program *prg";
+
+ for ( int p = 0; p < paramCount; p++ )
+ out << ", const char *p" << p;
+
+ out << " );\n";
+ }
+ }
+
+ out << "#endif\n";
+}
+
+void Compiler::generateExportsImpl()
+{
+ ostream &out = *outStream;
+
+ char *headerFn = strdup( exportHeaderFn );
+ char *suffix = strstr( headerFn, ".pack" );
+ if ( suffix != 0 && strcmp( suffix, ".pack" ) == 0 )
+ *suffix = 0;
+
+ if ( exportHeaderFn != 0 ) {
+ out << "#include \"" << headerFn << "\"\n";
+ }
+
+ out << "#include <colm/tree.h>\n";
+ out << "#include <string.h>\n";
+
+ /* Function implementations. */
+ for ( LelList::Iter lel = langEls; lel.lte(); lel++ ) {
+ if ( lel->objectDef != 0 ) {
+ FieldList &fieldList = lel->objectDef->fieldList;
+ for ( FieldList::Iter ofi = fieldList; ofi.lte(); ofi++ ) {
+ ObjectField *field = ofi->value;
+ if ( field->useOffset() && field->typeRef != 0 ) {
+ UniqueType *ut = field->typeRef->resolveType( this );
+
+ if ( ut != 0 && ut->typeId == TYPE_TREE ) {
+ out << ut->langEl->refName << " " << lel->declName <<
+ "::" << field->name << "() { return " <<
+ ut->langEl->refName << "( __prg, colm_get_attr( __tree, " <<
+ field->offset << ") ); }\n";
+ }
+ }
+
+ if ( field->isRhsGet() ) {
+ UniqueType *ut = field->typeRef->resolveType( this );
+
+ if ( ut != 0 && ut->typeId == TYPE_TREE ) {
+ out << ut->langEl->refName << " " << lel->declName <<
+ "::" << field->name << "() { static int a[] = {";
+
+ /* Need to place the array computing the val. */
+ out << field->rhsVal.length();
+ for ( Vector<RhsVal>::Iter rg = field->rhsVal; rg.lte(); rg++ ) {
+ out << ", " << rg->prodEl->production->prodNum;
+ out << ", " << rg->prodEl->pos;
+ }
+
+ out << "}; return " << ut->langEl->refName <<
+ "( __prg, colm_get_rhs_val( __prg, __tree, a ) ); }\n";
+ }
+ }
+ }
+ }
+ }
+
+ out << "\n";
+
+ for ( FieldList::Iter of = globalObjectDef->fieldList; of.lte(); of++ ) {
+ ObjectField *field = of->value;
+ if ( field->isExport ) {
+ UniqueType *ut = field->typeRef->resolveType(this);
+ if ( ut != 0 && ut->typeId == TYPE_TREE ) {
+ out <<
+ ut->langEl->refName << " " << field->name << "( colm_program *prg )\n"
+ "{ return " << ut->langEl->refName << "( prg, colm_get_global( prg, " <<
+ field->offset << ") ); }\n";
+ }
+ }
+ }
+
+ out << "\n";
+
+ for ( FunctionList::Iter func = functionList; func.lte(); func++ ) {
+ if ( func->exprt ) {
+ char *refName = func->typeRef->uniqueType->langEl->refName;
+ int paramCount = func->paramList->length();
+ out <<
+ refName << " " << func->name << "( colm_program *prg";
+
+ for ( int p = 0; p < paramCount; p++ )
+ out << ", const char *p" << p;
+
+ out << " )\n"
+ "{\n"
+ " int funcId = " << func->funcId << ";\n"
+ " const char *params[" << paramCount << "];\n";
+
+ for ( int p = 0; p < paramCount; p++ )
+ out << " params[" << p << "] = p" << p << ";\n";
+
+ out <<
+ " return " << refName <<
+ "( prg, colm_run_func( prg, funcId, params, " << paramCount << " ));\n"
+ "}\n";
+ }
+ }
+}