This is a project generator that uses Go and its package management for C++ packages. The structure of packages are defined in Go and files can be generated for Visual Studio (.sln, .vcxproj and .filters), Make, CMake, Xcode as well as Tundra. Work in progress for also supporting Zig as a build system.
If you like my work and want to support me. Please consider to buy me a coffee!
Any C++ external dependency like Boost, DirectX or whatnot should be wrapped in a package (github or other git server). There are a couple of notable features that can be triggered when generating the project files:
- generating
.clang-format
and/or.gitignore
- generating
source/test/cpp/test_main.cpp
- converting any file in
embedded
(following the directory structure) to C style array's so as to embed those files into your library/app - code generation of C++ enum from
.json
file
Also this project is a personal project and thus is not perfect but it serves my needs, feel free to post issues/requests if you want to see additional features.
This allows me to write packages (C++ libraries) and use them in another C++ package by defining a dependency on them. Using the go package management solution you can 'get' these packages and then by running go run %name%.go
you can generate solution and project files. The goal is to support the following IDE's and/or build-systems:
- Visual Studio (supported, Windows)
- Xcode (supported, Mac)
- Tundra (supported, Mac, Linux and Windows)
- CMake (supported, Mac, Linux and Windows)
- Make (supported, Mac and Linux, untested on Windows)
- Zig (WIP)
Note: Not happy with CMake, there must be a better way of doing this, if you know how please let me know.
Currently the design is quite set and the goal is to keep API changes to a minimum.
If you have repository/package that uses ccode, you can do the following to generate the make build files, this example uses the cbase
repository:
go run cbase.go --dev=make
- cd into
target/make
make
(will build all configuration, e.g. debug and release)make clean
(will clean all artifacts)make debugtest
(will build onlydebugtest
configuration)
For Visual Studio build files (on Windows, Visual Studio is the default generator):
go run cbase.go --dev=vs2022
- cd into
target/msdev
- You now should have Visual Studio solution and project files
For Tundra build files (on Mac and Linux, Tundra is the default generator):
go run cbase.go --dev=tundra
- cd into
target/tundra
- run
tundra2 debug
ortundra2 release
These are the steps to make a new package, or take a peek at one of my libraries,
like github.com/jurgen-kluft/cbase
:
- Create a new Github repository like
mylibrary
- In the root create a
mylibrary.go
file - In the root create a folder called
package
with a file in it calledpackage.go
- Once you have specified everything in package.go:
- In the root 'go get' (this will get all your specified dependencies in GO_PATH)
- To generate the VS solution (default on Windows) and projects just run:
go run mylibrary.go
- To generate the Tundra build file (default on MacOS) run:
go run mylibrary.go
Example:
The content of the mylibrary.go
file:
package main
import (
"github.com/jurgen-kluft/mylibrary/package"
"github.com/jurgen-kluft/ccode"
)
func main() {
ccode.Init()
// This will generate
// - ./.gitignore
// - ./.clang-format
// - ./source/test/cpp/test_main.cpp
ccode.GenerateFiles()
// This will generate the Visual Studio solution and projects, makefile, or tundra build files
ccode.Generate(mylibrary.GetPackage())
// You can also insert generated C++ enums with ToString and other functions, the my_enums.h
// file should already exist and have 2 delimiter lines that you can configure as
// 'between' (take a peek inside the `embedded/my_enums.h.json` file)
ccode.GenerateCppEnums("embedded/my_enums.h.json", "main/include/cbase/my_enums.h")
// Or if you are up to it, even generating structs is (wip) possible
ccode.GenerateCppStructs("embedded/my_structs.h.json", "main/include/cbase/my_structs.h")
}
The content of the /package/package.go
file with one dependency on 'myunittest':
package mylibrary
import (
"github.com/jurgen-kluft/ccode/denv"
"github.com/githubusername/myunittest/package"
)
// GetPackage returns the package object of 'mylibrary'
func GetPackage() *denv.Package {
// Dependencies
unittestpkg := myunittest.GetPackage()
// The main (mylibrary) package
mainpkg := denv.NewPackage("mylibrary")
mainpkg.AddPackage(unittestpkg)
// 'mylibrary' library
mainlib := denv.SetupDefaultCppLibProject("mylibrary", "github.com/githubusername/mylibrary")
mainlib.Dependencies = append(mainlib.Dependencies, unittestpkg.GetMainLib())
// 'mylibrary' unittest project
maintest := denv.SetupDefaultCppTestProject("mylibrary_test", "github.com/githubusername/mylibrary")
mainpkg.AddMainLib(mainlib)
mainpkg.AddUnittest(maintest)
return mainpkg
}
There are some requirements for the layout of folders inside of your repository to hold the library and unittest files, this is the layout:
source\main\cpp
: the cpp files of your library. Header files should be included as#include "mylibrary/header.h"
source\main\include\mylibrary
: the header files of your librarysource\test\cpp
: the cpp files of your unittest appsource\test\include
: the header files of your unittest appembedded\**
: all the files that need to be auto embedded or are used for code generation- binary file to .cpp
C array
- C++ enum code generation (from
.json
file)
- binary file to .cpp