import java.io.OutputStreamWriter; import java.util.stream.Collectors; /** * Write enum declarations and accessible value into C++ format. */ public class CppWriter { // =========== C++ Enum Declaration Writer =========== /** * Get corredponding C++ underlying type of given enum. *

* This is C++ specific function. * * @param canUnsigned The parameter stored in Enum_t that indicate whether this * enum can use unsigned int as its underlying type. * @return The string form of its underlying type. */ private static String getEnumUnderlyingType(boolean canUnsigned) { return canUnsigned ? "CKDWORD" : "CKINT"; } /** * Internal real enum declarations writer. * * @param writer {@linkplain java.io.OutputStreamWriter} instance for writing. * @param prog {@linkplain EnumsHelper.EnumCollection_t} instance for writing. * @throws Exception */ private static void internalWriteEnums(OutputStreamWriter writer, EnumsHelper.EnumCollection_t prog) throws Exception { IndentHelper indent = new IndentHelper(writer, CommonHelper.LangType.Cpp); for (EnumsHelper.Enum_t enum_t : prog.mEnums) { // write enum comment indent.briefComment(enum_t.mEnumComment); // write enum start indent.printf("enum class %s : %s {", enum_t.mEnumName, getEnumUnderlyingType(enum_t.mCanUnsigned)); indent.inc(); // write enum entries for (EnumsHelper.EnumEntry_t enumEntry_t : enum_t.mEntries) { // write entry self if (enumEntry_t.mEntryValue == null) { indent.printf("%s,", enumEntry_t.mEntryName); } else { indent.printf("%s = %s,", enumEntry_t.mEntryName, enumEntry_t.mEntryValue); } // write entry comment after member indent.afterMemberComment(enumEntry_t.mEntryComment); } // write enum tail indent.dec(); indent.puts("};"); } } /** * Write an enum collection into given file. *

* Actually this is a wrapper of internal enum collection writer. * * @param filename The name of written file. * @param prog {@linkplain EnumsHelper.EnumCollection_t} instance for * writing. * @throws Exception */ public static void writeEnums(String filename, EnumsHelper.EnumCollection_t prog) throws Exception { // open file and write OutputStreamWriter fs = CommonHelper.openOutputFile(filename); internalWriteEnums(fs, prog); fs.close(); } /** * Write a single enum into given file. *

* Actually this is a wrapper of internal enum collection writer. * * @param filename The name of written file. * @param _enum {@linkplain EnumsHelper.Enum_t} instance for writing. * @throws Exception */ public static void writeEnum(String filename, EnumsHelper.Enum_t _enum) throws Exception { // create an collection from single enum declaration // for suit the argument requirement of real writer. EnumsHelper.EnumCollection_t col = new EnumsHelper.EnumCollection_t(); col.mEnums.add(_enum); // open file and write OutputStreamWriter fs = CommonHelper.openOutputFile(filename); internalWriteEnums(fs, col); fs.close(); } // =========== C++ Enum Accessible Value Writer =========== /** * Internal real enum collection accessible value writer. * * @param writer {@linkplain java.io.OutputStreamWriter} instance for writing. * @param prog {@linkplain EnumsHelper.EnumCollection_t} instance for writing. * @param parts The part of these enum declarations. It will indicate the * namespace where find given enum collection. * @throws Exception */ private static void internalWriteAccVals(OutputStreamWriter writer, EnumsHelper.EnumCollection_t prog, CommonHelper.CKParts parts) throws Exception { IndentHelper indent = new IndentHelper(writer, CommonHelper.LangType.Cpp); // write type defination (just to let user know what the type is) indent.puts("// struct GeneralReflection { const char* mName; };"); indent.puts("// template"); indent.puts("// using GeneralReflectionArray = std::vector>;"); indent.puts(""); indent.puts(""); indent.puts(""); // write declarations for (EnumsHelper.Enum_t enum_t : prog.mEnums) { indent.printf("extern const GeneralReflectionArray %s;", CommonHelper.getCKPartsNamespace(parts), enum_t.mEnumName, enum_t.mEnumName); } indent.puts(""); indent.puts(""); indent.puts(""); // write implements for (EnumsHelper.Enum_t enum_t : prog.mEnums) { // write enum desc header indent.printf("const GeneralReflectionArray %s {", CommonHelper.getCKPartsNamespace(parts), enum_t.mEnumName, enum_t.mEnumName); indent.inc(); // write enum desc entries for (EnumsHelper.EnumEntry_t enumEntry_t : enum_t.mEntries) { indent.printf("{ LibCmo::%s::%s::%s, {\"%s\"} },", CommonHelper.getCKPartsNamespace(parts), enum_t.mEnumName, enumEntry_t.mEntryName, enumEntry_t.mEntryName); } // write enum tail indent.dec(); indent.puts("};"); } } /** * Write an enum collection accessible value into given file. *

* Actually this is a wrapper of internal enum collection accessible value * writer. * * @param filename The name of written file. * @param prog {@linkplain EnumsHelper.EnumCollection_t} instance for * writing. * @param parts The part of these enum declarations. * @throws Exception */ public static void writeAccVals(String filename, EnumsHelper.EnumCollection_t prog, CommonHelper.CKParts parts) throws Exception { // open file and write OutputStreamWriter fs = CommonHelper.openOutputFile(filename); internalWriteAccVals(fs, prog, parts); fs.close(); } /** * Write a single enum accessible value into given file. *

* Actually this is a wrapper of internal enum collection accessible value * writer. * * @param filename The name of written file. * @param _enum {@linkplain EnumsHelper.Enum_t} instance for writing. * @param parts The part of these enum declarations. * @throws Exception */ public static void writeAccVal(String filename, EnumsHelper.Enum_t _enum, CommonHelper.CKParts parts) throws Exception { // create a enum collection to fulfill the requirement of internal writer. EnumsHelper.EnumCollection_t col = new EnumsHelper.EnumCollection_t(); col.mEnums.add(_enum); // open file and write OutputStreamWriter fs = CommonHelper.openOutputFile(filename); internalWriteAccVals(fs, col, parts); fs.close(); } // =========== Specialized C++ Enum Accessible Value Writer =========== // Only accessible value part of CERROR and CK_CLASSID need to be specialized. // The enum self do not need special treat. Just write them normally. /** * Specialized CKERROR accessible value writer. *

* The declaration of CKERROR do not need special treat. It is okey to use * common writer. * * @param filename The name of written file. * @param errors The {@linkplain EnumsHelper.Enum_t} instance storing CKERROR. * @throws Exception */ public static void writeCkErrorAccVal(String filename, EnumsHelper.Enum_t errors) throws Exception { OutputStreamWriter writer = CommonHelper.openOutputFile(filename); IndentHelper indent = new IndentHelper(writer, CommonHelper.LangType.Cpp); // write type defination (just to let user know what the type is) indent.puts("// struct CkErrorReflection { const char* mName; const char* mDescription; };"); indent.puts("// using CkErrorReflectionArray = std::vector>;"); indent.puts(""); indent.puts(""); indent.puts(""); // write implementation indent.puts("const CkErrorReflectionArray CKERROR {"); indent.inc(); for (EnumsHelper.EnumEntry_t entry : errors.mEntries) { String comment = CommonHelper.escapeString(entry.mEntryComment); if (comment == null) comment = ""; indent.printf("{ LibCmo::CK2::CKERROR::%s, { \"%s\", \"%s\" } },", entry.mEntryName, entry.mEntryName, comment); } indent.dec(); indent.puts("};"); writer.close(); } /** * Specialized CK_CLASSID accessible value writer. *

* The declaration of CK_CLASSID do not need special treat. It is okey to use * common writer. * * @param filename The name of written file. * @param classids The {@linkplain EnumsHelper.Enum_t} instance storing * CK_CLASSID. * @throws Exception */ public static void writeCkClassidAccVal(String filename, EnumsHelper.Enum_t classids) throws Exception { OutputStreamWriter writer = CommonHelper.openOutputFile(filename); IndentHelper indent = new IndentHelper(writer, CommonHelper.LangType.Cpp); // write type defination (just to let user know what the type is) indent.puts("// struct CkClassidReflection { std::vector mHierarchy; };"); indent.puts("// using CkClassidReflectionArray = std::vector>;"); indent.puts(""); indent.puts(""); indent.puts(""); indent.puts("const CkClassidReflectionArray CK_CLASSID {"); indent.inc(); for (EnumsHelper.EnumEntry_t entry : classids.mEntries) { EnumsHelper.EnumEntryWithHierarchy_t specialized = (EnumsHelper.EnumEntryWithHierarchy_t) entry; String hierarchy = specialized.mHierarchy.stream().map(value -> value.mEntryName) .collect(Collectors.joining("\", \"")); indent.printf("{ LibCmo::CK2::CK_CLASSID::%s, { { \"%s\" } } },", entry.mEntryName, hierarchy); } indent.dec(); indent.puts("};"); writer.close(); } }