diff --git a/.github/workflows/trickops.yml b/.github/workflows/trickops.yml index cadfc5c4..d7a8dc8a 100644 --- a/.github/workflows/trickops.yml +++ b/.github/workflows/trickops.yml @@ -8,15 +8,15 @@ defaults: jobs: trickops-tests-ubuntu: - name: Unit Tests Ubuntu:20.04 - runs-on: ubuntu-20.04 - container: ubuntu:20.04 + name: Unit Tests Ubuntu:22.04 + runs-on: ubuntu-22.04 + container: ubuntu:22.04 steps: - uses: actions/checkout@master - name: install dependencies # Note that perl is for trick-gte which TrickOps runs and qt and everything after it is for koviz run: | - export DEBIAN_FRONTEND=noninteractive && apt-get update && apt-get install -y git python3 python3-venv perl perl-modules-5.30 qtbase5-dev wget unzip g++ make flex bison + export DEBIAN_FRONTEND=noninteractive && apt-get update && apt-get install -y git python3 python3-venv perl perl-modules-5.34 qtbase5-dev wget unzip g++ make flex bison - name: create virtual environment run: | cd share/trick/trickops/ @@ -41,7 +41,7 @@ jobs: trickops-tests-rockylinux8: name: Unit Tests RockyLinux8 - runs-on: ubuntu-20.04 + runs-on: ubuntu-22.04 container: rockylinux:8 steps: - uses: actions/checkout@master diff --git a/README.md b/README.md index e9801654..9731c538 100644 --- a/README.md +++ b/README.md @@ -16,9 +16,6 @@ 32-bit - -Coverage Status -

diff --git a/docs/developer_docs/DesConvertSwig.md b/docs/developer_docs/DesConvertSwig.md index 4264384c..c66f9310 100644 --- a/docs/developer_docs/DesConvertSwig.md +++ b/docs/developer_docs/DesConvertSwig.md @@ -275,10 +275,13 @@ Add the class_string to the SWIG interface text. 1. While there's class content text remaining to be processed, repeatedly search for data members that match : template_name '<' template-params '>' name ; For each match, create a SWIG %template directive to create an instantiation -of the specific templated type used by the data member. Add the -SWIG %template directive to the templated typedefs string that -Otherwise append whatever wasn't matched in process contents to -the SWIG interface text. +of the specific templated type used by the data member. Due to changes in SWIG 4, +template directives must be specified before their respective data members. As such, +the template directives are inserted immediately prior to the class definition. +SWIG does not resolve namespaces in these directives the same as C++ does +(it will only check local scope, not global). To account for this, +if the directive specifies a template outside the local namespace, convert_swig will +escape the current namespace in the format : } template_directive namespace name { ## process_typedef_struct diff --git a/docs/documentation/install_guide/Install-Guide.md b/docs/documentation/install_guide/Install-Guide.md index 32c7e94d..68ec426f 100644 --- a/docs/documentation/install_guide/Install-Guide.md +++ b/docs/documentation/install_guide/Install-Guide.md @@ -190,7 +190,7 @@ apt-get update apt-get install -y bison clang flex git llvm make maven swig cmake \ curl g++ libx11-dev libxml2-dev libxt-dev libmotif-common libmotif-dev \ python3-dev zlib1g-dev llvm-dev libclang-dev libudunits2-dev \ -libgtest-dev openjdk-11-jdk zip +libgtest-dev default-jdk zip # On some versions of Ubuntu (18.04 as of 04/2021), there may be multiple installations of python. # Our new python3-dev will be linked to python3 and python3-config in your bin. @@ -198,6 +198,8 @@ libgtest-dev openjdk-11-jdk zip export PYTHON_VERSION=3 ``` +Note: If you need to use a specific JDK version, such as `openjdk-11-jdk`, you can replace `default-jdk` with `openjdk-11-jdk` under install packages as shown above. However, you need to check where the `java` and `javac` commands are located. For instance, Ubuntu 24 typically sets up JRE (21) headless by default, so the `java` (version 21 headless) command might be located in `/usr/bin`. When you install `openjdk-11-jdk`, both `java` (version 11) and `javac` (version 11) might be placed in `/usr/lib/jvm/java-11-openjdk-amd64/bin`, with only `javac` potentially also in `/usr/bin`. Consequently, running a Java GUI with the default PATH might use JRE 21 headless instead of JRE 11, even though you’re using JDK 11 for compiling, which may not be the desired configuration. Placing `/usr/lib/jvm/java-11-openjdk-amd64/bin` before `/usr/bin` in your PATH ensures that only JDK 11 is used. + proceed to [Install Trick](#install) section of the install guide --- @@ -348,9 +350,9 @@ proceed to [Install Trick](#install) section of the install guide ### Build Clang and LLVM -#### If you come to this section because Clang+LLVM installed by the package manager on your machine does not work for your environment, you need to manually build Clang and LLVM. Following instructions show steps on building a particular release of Clang and LLVM . `cmake` is required. CMake may support multiple native build systmes on certain platforms. A generator is responsible for generating a particular build system. Below lists two approaches for your reference. The 1st approach uses `Unix Makefiles` (one of Makefile generators) and the 2nd one uses `Ninja` (one of Ninja generators). For Mac Apple Silicon user, may want to go to the 2nd approach direcly. +If you come to this section because Clang+LLVM installed by the package manager on your machine does not work for your environment, you need to manually build Clang and LLVM. Following instructions show steps on building a particular release of Clang and LLVM . `cmake` is required. CMake may support multiple native build systmes on certain platforms. A generator is responsible for generating a particular build system. Below lists two approaches for your reference. The first approach uses `Unix Makefiles` (one of Makefile generators) and the second one uses `Ninja` (one of Ninja generators). For Mac Apple Silicon user, may want to go to the second approach directly. -#### Note: Remember to add `--with-llvm=` for Trick configure if using the Clang and LLVM built in this section. +Note: Remember to add `--with-llvm=` for Trick configure if using the Clang and LLVM built in this section. 1. Using `Unix Makefiles` generator diff --git a/docs/documentation/running_a_simulation/Input-File.md b/docs/documentation/running_a_simulation/Input-File.md index 4410b9db..1153cbe2 100644 --- a/docs/documentation/running_a_simulation/Input-File.md +++ b/docs/documentation/running_a_simulation/Input-File.md @@ -148,9 +148,11 @@ Structure and class variables do not carry around units, and therefore the units ## Creating New Objects and Allocating Memory It is possible to create new objects and allocate new memory for structures directly in the Python -input file. There are at least two ways to allocate memory. +input file. Three different ways are described below. -The first method is to call the `Trick::MemoryManager` routines to allocate memory. This is the preferred method. +### 1. Call ```Trick::MemoryManager``` Allocation Routines Directly + +The first method is to call the `Trick::MemoryManager` routines to allocate memory. There are 3 `Trick::MemoryManager` calls with varying arguments that can be used to allocate memory ```python @@ -188,7 +190,80 @@ food[2] = trick.alloc_type(6, "double") Memory allocated using the above routines are tracked by the memory manager and is checkpointable and data recordable. -The second method is to call the wrapped constructor of the class directly. This is analogous to declaring local +### 2. Use a Factory Function +The benefit of this method is flexibility in how objects are initialized. For example, we might want to initialize our objects with a non-default constructor. So, the requirements for our factory function are: + +1. Allocate a memory object via the Trick Memory Manager, and +2. Call a constructor to initialize the object (using placement-new) + +#### A Few Words About Placement-new +In C++ one often instanciates a class object using the **new** operator, for example: + +``` +MyClass * p = new MyClass(a,b,c); +``` + +This form of **new** + +1. allocates memory, and then +2. calls a constructor. + +Another form of **new**, is called "placement-new". Rather than allocating and calling a constructor to initialize memory, placement-new simply calls a constructor to initialize memory that has already been allocated (e.g., from the Memory Manager). + +If ```p``` points to allocated memory, then we can initialize that memory with **placement-new** : + +``` +new (p) MyClass(a,b,c); +``` + +In our factory function we'll pass the object pointer we got from the Memory Manager to our placement-new call, to initialize it. + +#### Example Factory Function from ```SIM_contact``` +SIM_contact simulates collisions between moving balls (think "pool balls"). From the input file, one or more balls can be added to the simulation. Each ball is initialized with a mass, a size, a position, and a velocity. The Ball class also includes a (non-default) constructor. + +```C++ +class Ball { + public: + Ball(){} + double pos[2]; + double vel[2]; + double mass; + double radius; + + // A Non-Default Constructor + Ball(double x, double y, double vx, double vy, double r, double m); +}; +``` +To create and initialize a new Ball object, we have the function ```make_Ball```. + +```C++ +// Factory function Implementation +Ball* make_Ball(double x, double y, double vx, double vy, double r, double m) { + Ball* b = (Ball*)TMM_declare_var_s("Ball"); + return (new (b) Ball(x,y,vx,vy,r,m)); +} +``` + +Because this function is bound to Python by SWIG, it can be called from the input file. +For example : + +##### From ```RUN_Newtons_cradle/input.py``` +```Python +dyn.contact.nballs = 7 +dyn.contact.balls = trick.TMM_declare_var_1d("Ball*", dyn.contact.nballs) +dyn.contact.balls[0] = trick.make_Ball(-4.00, 0.0, 2.0, 0.0, 0.5, 1.0) +dyn.contact.balls[1] = trick.make_Ball(-1.00, 0.0, 0.0, 0.0, 0.5, 1.0) +dyn.contact.balls[2] = trick.make_Ball( 0.01, 0.0, 0.0, 0.0, 0.5, 1.0) +dyn.contact.balls[3] = trick.make_Ball( 1.02, 0.0, 0.0, 0.0, 0.5, 1.0) +dyn.contact.balls[4] = trick.make_Ball( 2.03, 0.0, 0.0, 0.0, 0.5, 1.0) +dyn.contact.balls[5] = trick.make_Ball( 7.00, 0.0, 0.0, 0.0, 1.0, 1000000.0) +dyn.contact.balls[6] = trick.make_Ball(-7.00, 0.0, 0.0, 0.0, 1.0, 1000000.0) +``` + +This creates and initializes seven *Ball* objects needed to configure a Newton's cradle. + +### 3. Call the Wrapped Class Constructor Directly +The third method is to call the wrapped constructor of the class directly. This is analogous to declaring local variables in C/C++ routines. And like local variables in C/C++ if the python variable goes out of scope in the input file, then python will try and free the memory associated with the local object. Memory allocated this way is not checkpointable or data recordable. diff --git a/docs/documentation/running_a_simulation/Running-a-Simulation.md b/docs/documentation/running_a_simulation/Running-a-Simulation.md index 7baaef8e..b8e9fc56 100644 --- a/docs/documentation/running_a_simulation/Running-a-Simulation.md +++ b/docs/documentation/running_a_simulation/Running-a-Simulation.md @@ -16,14 +16,27 @@ S_main_${TRICK_HOST_CPU}.exe [trick_version] [sie] [-u ] ``` -- The first argument in the command line must be the simulation input file name. The input file name can be in the form of a full path name but MUST have a RUN_ directory immediately above the input file name. By default, all the simulation output is directed to this RUN_ directory. The standard is input.py; however, a simulation could be started from a checkpoint file by substituting chkpnt_