Difference between revisions of "Compiling in MacOS X"

From OSUPDOCS
Jump to navigation Jump to search
 
(229 intermediate revisions by the same user not shown)
Line 3: Line 3:
== Using XCode ==
== Using XCode ==


Most development of [[NairnMPM]] and [[NairnFEA]] is done in MacOS X and thus compiling on MacOS X is easy. The preferred method is to use XCode (but first you must [[#Installing XCode|install Xcode, install command line tools, and get a new compiler]]). A complete XCode project is located at
Most development of [[NairnMPM]] and [[NairnFEA]] is done in MacOS X and thus compiling on MacOS X should be easy. The preferred method is to use XCode (but first you must [[#Installing XCode|install XCode, install command line tools, and pick a compiler]]). A complete XCode project is located at


  nairn-mpm-fea/Common/Projects/NairnMPM.xcodeproj
  nairn-mpm-fea/Common/Projects/NairnMPM.xcodeproj


This project is called <tt>NairnMPM</tt>, but it includes all MPM and FEA source code in two targets named <tt>NairnMPM</tt> and <tt>NairnFEA</tt>. Once [[#Compiling Xerces on MacOS X|<tt>xerces</tt> is installed]], open the <tt>NairnMPM.xcodeproj</tt> to compile the code by:
This project is called <tt>NairnMPM</tt>, but it includes all MPM and FEA source code in two targets named <tt>NairnMPM</tt> and <tt>NairnFEA</tt> (as well as targets for [[ExtractMPM]] and [[CompareGlobal]]). Once [[#Compiling Xerces on MacOS X|<tt>xerces</tt> is installed]], open the <tt>NairnMPM.xcodeproj</tt> to compile the code by:


# Select the target to compile - either <tt>NairnMPM</tt> or <tt>NairnFEA</tt>
# Select the target to compile - either <tt>NairnMPM</tt> or <tt>NairnFEA</tt> (or [[ExtractMPM]] or [[CompareGlobal]])
# Choose build and they will compile and be saved in XCode's derived data folder.
# Choose build and they will compile and be saved in XCode's derived data folder.


If a linking error occurs, you might have conflicting architectures between the XCode settings and the [[#Compiling Xerces on MacOS X|<tt>xerces</tt> library you installed]]. To fix this problem edit <i>both</i> the project and target settings and under the &quot;Architectures&quot; section, set the &quot;Architectures&quot; option to match the architecture you used when [[#Compiling Xerces on MacOS X|installing <tt>xerces</tt>]] or recompile <tt>xerces</tt> if needed.
If a compile error occurs, you might need to [[#XCode Build Settings|change project settings]] to match you method for [[#Compiling for OpenMP|compiling for OpenMP]].
 
If a linking error occurs, you might have conflicting architectures between the XCode settings and the [[#Compiling Xerces on MacOS X|<tt>xerces</tt> library you installed]] or be missing some important C++ libraries. To fix <tt>xerces</tt> problems edit <i>both</i> the project and target settings and under the &quot;Architectures&quot; section, set the &quot;Architectures&quot; option to match the architecture you used when [[#Compiling Xerces on MacOS X|installing <tt>xerces</tt>]] or recompile <tt>xerces</tt> if needed. C++ library problems are sometimes caused by MacOS updates and might need some new [[#XCode Build Settings|project settings]].


== Compiling Xerces on MacOS X ==
== Compiling Xerces on MacOS X ==
Line 24: Line 26:
  /usr/local/include
  /usr/local/include


for the header files. These can be changed if needed by editing the project and target settings. The <tt>nairn-mpm-fea</tt> project needs access to <tt>xerces 3.0</tt> or newer. This libraries can be obtained by downloading the  
for the header files. These can be changed if needed by editing the [[#XCode Build Settings|project and target settings]]. The <tt>nairn-mpm-fea</tt> project needs access to <tt>xerces 3.0</tt> or newer. This libraries can be obtained by downloading the  
[http://xml.apache.org/xerces-c/index.html xerces source code] from the [http://www.apache.org/ Apache Software Foundation] web site and then building and installing <tt>xerces</tt> with the following steps:
[http://xml.apache.org/xerces-c/index.html xerces source code] from the [http://www.apache.org/ Apache Software Foundation] web site and then building and installing <tt>xerces</tt> with the following steps:


<ol>
<ol>
<li>First [[#Installing XCode|install XCode and command line tools]].</li>
<li>First [[#Installing XCode|install XCode and command line tools]].</li>
<li>Open Terminal app and navigate to the xerces source folder expanded from the downloaded file.</li>
<li>Open Terminal app and navigate to the xerces folder expanded from the downloaded file.</li>
<li>Configure the code with the command:
<li>Configure the code with the command:
<pre>./configure CFLAGS="-arch x86_64 -arch arm64" CXXFLAGS="-arch x86_64 -arch arm64"</pre>
<pre>./configure CFLAGS="-arch x86_64 -arch arm64" CXXFLAGS="-arch x86_64 -arch arm64"</pre>
Line 43: Line 45:
== MacOS X Command Line Compiling ==
== MacOS X Command Line Compiling ==


It also possible to compile on MacOS X using a command line approach (after [[#Installing XCode|installing XCode and command line tools]] and after installing <tt>xerces</tt> and its header files as [[#Compiling Xerces on MacOS X|explained above]] making sure they are in the specified default locations). You can compile  [[NairnMPM]] using:
It also possible to compile on MacOS X using a command line approach. After [[#Installing XCode|installing XCode, command line tools, and a compiler]] and after [[#Compiling Xerces on MacOS X|installing <tt>xerces</tt> and its header files]], you can compile  [[NairnMPM]] using:


<pre>cd nairn-mpm-fea/NairnMPM/build
<pre>cd nairn-mpm-fea/NairnMPM/build
make -j 8 SYSTEM=mac-clang
make -j 8 SYSTEM=mac (or SYSTEM=mac-clang)
</pre>
</pre>


Line 52: Line 54:


<pre>cd nairn-mpm-fea/NairnFEA/build
<pre>cd nairn-mpm-fea/NairnFEA/build
make -j 8 SYSTEM=mac-clang
make -j 8 SYSTEM=mac (or SYSTEM=mac-clang)
</pre>
</pre>


The <tt>-j 8</tt> option compiles in parallel where the number is number of available cores on your computer.
The <tt>-j 8</tt> option compiles in parallel where the number is number of available cores on your computer. The <tt>SYSTEM</tt> setting tells the <tt>make</tt> process what compiler to use. Use <tt>SYSTEM=mac</tt> if you have [[#Convince Apple-Installed clang to Compile for OpenMP|modified Apple's compiler]] or <tt>SYSTEM=mac-clang</tt> if you [[#Install clang-mp Compiler|installed a separate <tt>clang-mp</tt>]].
All source code will be compiled and the executables will be installed in <tt>nairn-mpm-fea/NairnMPM/input</tt> or <tt>nairn-mpm-fea/NairnFEA/input</tt>, respectively. You can use an additional <tt>make install</tt> command to copy each compiled executable to your <tt>~/bin</tt> folder if desired.
 
The above command will compile the source cade and the executables will be installed in <tt>nairn-mpm-fea/NairnMPM/input</tt> or <tt>nairn-mpm-fea/NairnFEA/input</tt>, respectively. You can use an additional <tt>make install</tt> command to copy each compiled executable to your <tt>~/bin</tt> folder if desired.
 
If the command-line compile does not work, the most likely explanations are:


The above commands assume you have [[#Install Clang-mp|installed the <tt>clang-mp</tt>]] compiler. If you prefer to [[#Install GCC 4.8 or Newer|install and use <tt>gcc</tt>]] change the <tt>SYSTEM=mac-clang</tt> lines to <tt>SYSTEM=mac</tt> instead.
# A problem with your compiler settings. Two options are to to edit the <tt>makefile</tt> to use your [[#Compiling for OpenMP|chosen compiler]] or to [[#Compiling Old Version of the Code|pass alternate parameters to the <tt>make</tt> command]].
# A problem with the <tt>xerces</tt> installation. You either have to install it as specified [[#Compiling Xerces on MacOS X|above]] or edit the <tt>makefile</tt> to recognize your custom installation. The process is documented in the <tt>makefile</tt> and involves editing the <tt>xercesSo</tt> and <tt>headersPath</tt> variables for your different settings.


If the command-line compile does not work, the most likely explanation is a problem with the <tt>xerces</tt> installation. You either have to install it as specified [[#Compiling Xerces on MacOS X|above]] or edit the <tt>makefile</tt> to recognize your custom installation. The process is documented in the <tt>makefile</tt> and involves editing the <tt>xercesSo</tt> and <tt>headersPath</tt> variables for your different settings.
=== Compiling Old Version of the Code ===


You can pass additional parameters to the <tt>make</tt> command to alter the compilation process (''e.g.'', a <tt>CC</tt> option can pick a compiler installed on your computer). See comments in the <tt>makefile</tt> for all the latest options.
You can change settings used in the <tt>make</tt> process without editing the <tt>makefile</tt> by passing alternate parameters to the <tt>make</tt> command. One reason to use this approach is to compile old versions of the code. If old versions are checked out, the XCode project may not work (because of changes in XCode versions, build settings, or compiler you have installed). When this occurs, you can use command-line compile methods instead. For example, if working with current code on on Mac and the clang-mp compiler, you may be able to compile old versions using the build folders and the command


== Installing XCode ==
<pre>make CC=/opt/local/bin/clang-mp-11 CCLINK=/opt/local/bin/clang++-mp-11 \
        xercesSo=/usr/local/lib/libxerces-c.dylib \
        CFLAGS="-c -O3 -fopenmp -std=c++11 -I/opt/local/include/libomp" \
        LFLAGS="-fopenmp -lc++"
</pre>


All compiling on Mac requires that you install Xcode (even if you do not plan to use it for anything else). You can get Xcode from the Mac App Store or from the Apple Developer site. It is a large install. Once installed, you then need to [[#Installing Command Line Tools|install command line tools]].
where <tt>CC</tt> and <tt>CCLINK</tt> refers to a <tt>clang-mp</tt> compiler and linker you have installed (old versions of this project do not use <tt>CCLINK</tt>). The other parameters might also need changing. You can open the project's <tt>makefile</tt> to see its default options for these variables and then change as needed.


Unfortunately, the compilers provided in Apple's "Command Line Tools" do not support OpenMP used to make these code engines parallel. The two supported solutions are to [[#Compile Without OpenMP|compile without OpenMP]] (and lose all advantages of multiprocessor execution) or to [[#Install Clang-mp|install and use the <tt>clang-mp</tt> compiler]]. Some notes below explain how to [[#Deprecated Compiler Options|install and use an unsupported GCC compiler]].
When working with current version of the project, a better approach than using command-line arguments is to edit the <tt>makefile</tt> and pick your own default settings. Support for using a [[#Convince Apple-Installed clang to Compile for OpenMP|modifed Apple compiler]] should be built in already using the <tt>SYSTEM=mac</tt> setting. To add support for a [[#Install clang-mp Compiler|different version of <tt>clang-mp</tt>]], just follow examples used for others versions in the current <tt>makefile</tt>.


The current check out is configured to compile in XCode using [[#Install Clang-mp|<tt>clang-mp-7.0</tt>]]. Prior check outs are set up to use earlier versions of <tt>clang-mp</tt> or  various version of GCC. It is be easy to convert any checkout to use the compiler you prefer by making a few changes in the [[#XCode Build Settings|XCode project build settings]]. Alternatively, you can use command line compilation and [[#Compiling Old Version of the Code|specify settings for currently installed compiler]]
== Installing XCode ==


Whenever you update your MacOS or Xcode, you may need to [[#Possible Changes Needed After Updating MacOS or XCode|repeat some installation steps]] before you can return to compiling the code engines.
The simplest way to compiling on Mac requires that you install XCode. You can get XCode from the Mac App Store or from the Apple Developer site. It is a large install. Once installed, you then need to [[#Installing Command Line Tools|install command line tools]]. Unfortunately, the compilers provided in Apple's "Command Line Tools" do not support OpenMP used to make these code engines parallel. To compile for parallel calculations (which is important), see the two options below to [[#Compiling for OpenMP|compile for OpenMP]]. Whenever you update your MacOS or XCode, you may need to [[#Possible Changes Needed After Updating MacOS or XCode|repeat some installation steps]] before you can return to compiling the code engines.


The following sections have more details on installations needed for compiling the code engines.
The following sections have more details on installations needed for compiling the code engines.
Line 78: Line 88:
=== Installing Command Line Tools ===
=== Installing Command Line Tools ===


Apple used to install command line tools by default, but stopped doing that in MacOS Lion. You now have to manually install the tools before you can compile code. These steps should work:
Apple used to install command line tools by default, but stopped doing that a long time ago. You now have to manually install the tools before you can compile code. These steps should work:


# After [[#Insetalling XCode|installing XCode]] on a recent MacOS, launch the app and agree to all options. Once that is done, install command line tools by opening Terminal app and enter: <tt>xcode-select --install</tt>. If this step does not work, you can see [http://guide.macports.org/#installing.xcode MacPorts] for possible new instructions. You can also get the command line tools from the [https://developer.apple.com/downloads/index.action Apple developer website] and search for command line tools (being sure to get the correct tools for you current Xcode and MacOS versions). You may need to be signed up as an Apple developer for this method.
<ol>
# You may need to agree to XCode license by using Terminal app to enter: <tt>sudo xcodebuild -license</tt>
<li>The simplest process is to open the Terminal app and enter:
# Most tools should now be available. You should be able to make and install the [[#Compiling Xerces on MacOS X|<tt>xerces</tt>]] library.
<pre>xcode-select --install</pre>
# To compile the code engines for parallel code, you will additionally need to [[#Install Clang-MP|install a compiler]] that works with OpenMP (none is provided in the command line tools).
This install may or may not require XCode to already be installed. If it fails try installing XCode first. If it stall fails, refer to [http://guide.macports.org/#installing.xcode MacPorts documentation] or try installing [https://brew.sh/ HomeBrew] (which I think handles command-line too installation as well.</li>
<li>You may need to agree to XCode license by using Terminal app to enter <pre>sudo xcodebuild -license</pre></li>
<li>Most tools should now be available. You should be able to make and install the [[#Compiling Xerces on MacOS X|<tt>xerces</tt>]] library.</li>
<li>To compile the code engines for parallel code, you will additionally need to [[#Compiling for OpenMP|install a compiler]] that works with OpenMP (none is provided in the command line tools).</li>
</ol>


With each new system, Apple seems to creating road blocks to keep you from using you computer for interesting programming. For now, by using a few tricks it is still possible to get around their road blocks. Hopefully these tricks will continue to work in the future.
With each new system, Apple creates road blocks to keep you from using you computer for interesting programming. For now, by using a few tricks it is still possible to get around their road blocks. Hopefully these tricks will continue to work in the future.


One issue in the XCode project is that it might not find your <tt>xerces</tt> library, even if it is in the standard location. If it is not found, the library may be in <font color="red">red</font> in the &quot;External Frameworks and Libraries&quot; folder (or may not). A potential solution is to delete the reference to the library and then add it back. The problem is that you cannot navigate to the <tt>/usr/local/lib</tt> folder anymore (another new Apple &quot;feature&quot;). Here is a trick to get there:
One issue in the XCode project is that it might not find your <tt>xerces</tt> library, even if it is in the standard location. If it is not found, the library may be in <font color="red">red</font> in the &quot;External Frameworks and Libraries&quot; folder (or may not). A potential solution is to delete the reference to the library and then add it back. The problem is that you cannot navigate to the <tt>/usr/local/lib</tt> folder anymore (another new Apple &quot;feature&quot;). Here is a trick to get there:
Line 94: Line 108:
# Go through your new &quot;local&quot; folder and select the <tt>xerces</tt> library file in <tt>/usr/local/lib</tt>.
# Go through your new &quot;local&quot; folder and select the <tt>xerces</tt> library file in <tt>/usr/local/lib</tt>.


=== Install Clang-mp ===
=== Compiling for OpenMP ===


The currently recommended method to compile with OpenMP on a Mac and when using XCode is to install the <tt>clang-mp</tt> compiler . A good way is to install <tt>clang-mp</tt> is to use [http://www.macports.org MacPorts]. Once MacPorts tool is installed (and [[#Possible Changes Needed After Updating MacOS or XCode|possibly updated for current MacOS]]), the installation of <tt>clang-mp</tt> is easy. All you need is to open the Terminal app and enter:
Although Apple uses <tt>clang</tt> for compiling in XCode, for reasons that are hard to justify, they opted to disable compiling with OpenMP (<i>i.e.</i>, they disabled the <tt>-fopenmp</tt> compiler option). They further did not include <tt>libomp.dylib</tt>, which is needed to run software written for OpenMP. Because doing parallel computations is essential for large calculations, some method to use OpenMP is needed. Two possible solutions are given in this section.


sudo port selfupdate
==== Convince Apple-Installed <tt>clang</tt> to Compile for OpenMP ====
sudo port install clang-7.0 ld64 +ld64_xcode


The first command above is only needed to make sure your MacPorts is up to date. The second step will install the compiler at
The approach with minimal installation is to override Apple's disabling of the <tt>-fopenmp</tt> option to compile for OpenMP anyway. This approach uses the following steps:
 
/opt/local/bin/clang-mp-7.0
 
You can change to different version number if needed or when available. The checked-out XCode project may be setup for a different version of clang-mp. You can either switch to the desired version or change few [[#XCode Project Settings|XCode project settings]]. When switching to new version, you might want to remove the old version first with
 
sudo port uninstall --follow-dependents clang-7.0
 
Make sure to use same name of the port used when you installed. See [http://www.macports.org MacPorts] on removing and an old version and replacing with a newer one. A "clean" approach when upgrading to a new MacOS is:
 
# [https://guide.macports.org/chunked/installing.macports.uninstalling.html Uninsall all of MacPorts]
# Upgrade MacOS
# [https://guide.macports.org/chunked/installing.macports.html Install MacPorts for the new MacOS]
# Install all ports needed for this software.
 
If these are the only ports you need, the process is easy. If you install ports for other uses, you might want to uses one of the other approaches on the [http://www.macports.org MacPorts] web site when migrating new a new MacOS.
 
To use <tt>clang-mp</tt> in XCode, you will need to add some [[#XCode Build Settings|project build settings]]. For [[#MacOS X Command Line Compiling|command line compiling]], you may need to update the <tt>makefile</tt> to use you new compiler (see comments in <tt>makefile</tt> for details). Alternatively, you can use the <tt>CC</tt> option to specify path to any compiler, such as:
 
make SYSTEM=mac CC=/opt/local/bin/clang-mp-4.0
 
=== XCode Build Settings ===
 
Compiling using a <tt>clang-mp</tt> compiler in Xcode requires some specific project build settings. Various versions of the checkout may have settings for one specific compiler, but you should be able to easily switch to your preferred settings using the details below:


<ol>
<ol>
<li>Using <tt>clang-mp</tt>
<li>The only extra installation required is a signed copy of the <tt>libomp.dylib</tt>. Fortunately. a member of the R project has made this installation easy. First check your version of XCode. Next, look up file name you need in the following table:
<ul>
{| class="wikitable"
<li><b>Compiler for C/C++/Objective-C</b>: select the default compiler.</li>
|-
<li>Use menu command Editor&rarr;Add Build Setting&rarr;Add User-Defined Setting and add setting for <b>CC</b> with value <b>/opt/local/bin/clang-mp-7.0</b> (using version number you installed).</li>
! XCode Version !! File name to download
<li>Use menu command Editor&rarr;Add Build Setting&rarr;Add User-Defined Setting and add setting for <b>LDPLUSPLUS</b> with value <b>/opt/local/bin/clang++-mp-7.0</b> (using version number you installed).</li>
|-
<li><b>Other C Flags</b>: add <tt>-fopenmp</tt>.</li>
| align="center"  | 15.0+ || openmp-16.0.4-darwin20-Release.tar.gz
<li><b>Other C++ Flags</b>: add <tt>$(OTHER_CFLAGS)</tt> to have same flags used for C++ compile as well.</li>
|-
<li><b>Other Linker Flags</b>: add <tt>-fopenmp</tt>.</li>
| align="center"  | 14.3.x || openmp-15.0.7-darwin20-Release.tar.gz
<li><b>Enable Modules (C and Objective-C)</b>: set to <tt>No</tt></li>
|-
<li><b>Header Search Paths</b>: add <tt>/opt/local/include/libomp</tt>
| align="center"  | 14.0 to 14.2 || openmp-14.0.6-darwin20-Release.tar.gz
</ul>
|-
| align="center"  | 13.3 to 13.4.1 || openmp-13.0.0-darwin21-Release.tar.gz
|-
| align="center"  | 13.0 to 13.2.1 || openmp-12.0.1-darwin20-Release.tar.gz
|}
If you have a newer version of XCode, you can check the [https://mac.r-project.org/openmp/ R project posting] for a new file name. If you have an older version of XCode, now is a good time to update your Mac.</li>
<li>Once you have the file name, download it from <tt>r-project.org</tt> in the Terminal app using:
<pre>curl -O https://mac.r-project.org/openmp/openmp-15.0.7-darwin20-Release.tar.gz</pre>
replacing the file name with the one you need for your version of XCode. If you visit the [https://mac.r-project.org/openmp/ R project posting], do not download with the links in that web site because Apple will quarantine all files rendering the library unusable without more work.
</li>
</li>
<li>Compiling Without OpenMP
<li>Once downloaded, extract and install <tt>libomp</tt> files with
<ul>
<pre>sudo tar fvxz openmp-15.0.7-darwin20-Release.tar.gz -C /</pre>
<li><b>Compiler for C/C++/Objective-C</b>: select the default compiler.</li>
being sure to replace the file name in this command with the one downloaded in the previous step.</li>
<li><b>Other C Flags</b>: remove <tt>-fopenmp</tt></li>
<li>The <tt>-C /</tt> option in the previous step extracts the files directly to your root <tt>/usr/local</tt> folder rather a folder next to the downloaded file. When done, the following files should be installed:
<li><b>Other C++ Flags</b>: remove <tt>-fopenmp</tt></li>
<pre>/usr/local/lib/libomp.dylib
<li><b>Other Linker Flags</b>: none.</li>
/usr/local/include/ompt.h
<li>Comment out <tt>#define USE_OPENMP</tt> in the <tt>MPMPrefix.hpp</tt> and <tt>FEAPrefix.hpp</tt> files.</li>
/usr/local/include/omp.h
</ul>
/usr/local/include/omp-tools.h</pre>
</li>
</li>
</ol>
</ol>


Note that compiling without OpenMP will create serial code. Because it will be much slower than parallel code, that option is only recommended when no other option works (for some reason).
Once these steps are done, you are ready to [[#Using XCode|compile in XCode]] or [[#MacOS X Command Line Compiling|compile on a command line]] (using the <tt>SYSTEM=mac</tt> option). If an XCode compile has problems, you may have adjust some of the [[#Project Setting for Apple's Compiler|XCode project settings]] for this compiler option.


=== Possible Changes Needed After Updating MacOS or XCode ===
According to the [https://mac.r-project.org/openmp/ R project posting], this approach works now, but " may break at any time." Furthermore, each time you upgrade XCode, you may need to install a signed and updated version of <tt>libomp.dylib</tt>. If for some reason the method stops working or an updated <tt>libomp.dylib</tt> cannot be found, you can switch to an alternate method that involves installing a separate compiler (and is explain in the next section).


Unfortunately, Apple does not do a very good job of preserving your prior settings after a major upgrade and you will often need a complete reinstall of anything installed using MacPorts. Some things that might be needed with each MacOS upgrade are:
==== Install <tt>clang-mp</tt> Compiler ====


# Get the latest version of XCode (from developer web site or app store).
An alternate approach to compiling parallel code using OpenMP is to install a separate <tt>clang-mp</tt> compiler. One way to install this compiler is to use [http://www.macports.org MacPorts] as follows:
# Reinstall command line tools in Terminal app by entering <tt>xcode-select --install</tt>
# Agree to Xcode license in Terminal app using <tt>sudo xcodebuild -license</tt> (and provide your admin password).
# You may need to uninstall all installed MacPorts, reinstall MacPorts, and then reinstall all previously installed options. The process is explained here [https://trac.macports.org/wiki/Migration Migrating a MacPorts Installation]. Also see [[#Install Clang-mp|above section]] about migrating when only using ports for this software.
# To compile in XCode, you may need more changes (but hard to predict what Apple will require next).


=== Compiling Old Version of the Code ===
<ol>
<li>If not already installed, install [http://www.macports.org MacPorts]. Be aware that it might need to be [[#Possible Changes Needed After Updating MacOS or XCode|reinstalled after updating your MacOS]].</li>
<li>Open the Terminal app and enter
<pre>sudo port selfupdate
sudo port install clang-11 ld64 +ld64_xcode</pre>
The first command above is only needed to make sure your MacPorts is up to date (you might need to reinstall instead of <tt>selfupdate</tt>  [[#Possible Changes Needed After Updating MacOS or XCode|after updating your MacOS]]). The second step installs the compiler version requested (change "11" to version you want)
</li>
<li>To verify it installed, check in Terminal app:
<pre>which clang-mp-11
/opt/local/bin/clang-mp-11</pre>
If the <tt>which</tt> command did not return the installed location, an error occurred during install.
</ol>


If you need to compile old versions of the code, the XCode project may not work well (because of changes in XCode versions, build settings, or compiler you have installed). One option to compile old version (such as when searching for bugs) is to compile using command line and your currently installed compiler. For example, if working with current code on Mac and the clang-mp compiler, you may be able to compile old versions using the build folders and the command
The above install usually works. Some issues have occurred in MacOS Sonoma and new issues might happen in future MacOS versions. Here are some known issues:


<pre>make CC=/opt/local/bin/clang-mp-7.0 xercesSo=/usr/local/lib/libxerces-c.dylib
# A MacPorts <tt>install</tt> command in MacOS Sonoma reports that <tt>clang-mp</tt> versions 11, 12, and 13 are "known to fail." It appears these version can still be installed, but the <tt>sudo port install</tt> may not finish. If it exits without a binary, repeat the <tt>install</tt> command until it (hopefully) finishes. Each repeat picks up where it left off. If it hangs up while installing, force exit with Control-C and repeat again. If it eventually finishes, the install seems to work. I needed three repeats for one successful install of <tt>clang-mp-11</tt>.
        CFLAGS="-c -O3 -fopenmp -std=c++11 -I/opt/local/include/libomp"
# Installing <tt>clang-mp</tt> versions 14 or newer in MacOS Sonoma does not say "known to fail", but the first <tt>install</tt> command may exit with an unexpected "permission denied" error. Repeating the <tt>install</tt> command then picks up where it left off and finishes the install.
        LFLAGS="-fopenmp -lc++"
# Although these versions get installed, you may need to verify the [[#Project Settings for clang-mp|XCode project settings]] match both your installed version of <tt>clang-mp</tt> and your choice to use a separate compiler. A new issue occurs in MacOS Sonoma, but that is corrected with a new [[#Project Settings for clang-mp|runpath project setting]]. Future MacOS's might cause more problems and some might be fixable with new project settings.
</pre>


The above uses several lines, but it all should be on a single command line.
You can install multiple versions of <tt>clang-mp</tt>, which can be useful if working with multiple checkouts of this project. More likely, you only need one version. If you settle on one version, you can remove older ones with


== Deprecated Compiler Options ==
sudo port uninstall --follow-dependents clang-xx


Use of the <tt>GCC</tt> compiler is no longer recommended when working with XCode. It may not even work in XCode 8 or newer (at least by methods described below). The <tt>GCC</tt> compiler does work fine, however, if using only [[#MacOS X Command Line Compiling|command line methods]]. The following sections may provide useful information for installing it for that use.
where <tt>xx</tt> is the version you want to uninstall.


=== Install GCC 4.8 or Newer ===
=== XCode Build Settings ===


<tt>GCC 4.9</tt> or new supports OpenMP and may have some uses (<tt>GCC 4.8</tt> and older should be avoided). If you want to use it, a good way to install GCC 4.9 or newer is to use [http://www.macports.org MacPorts]. Once MacPorts tool is installed, the installation is explained in this [http://www.ficksworkshop.com/blog/14-coding/65-installing-gcc-on-mac blog]. In brief, all you need is to open the Terminal app and enter:
The downloaded XCode project will have project settings for compiling with [[#Convince Apple-Installed clang to Compile for OpenMP|Apple's compiler]] or with an [[#Install clang-mp Compiler|installed <tt>clang-mp</tt> compiler]]. To find out which one, open the project in XCode, find the "Build Settings" for the <tt>NairnMPM</tt> or <tt>NairnFEA</tt> target, and scroll the to bottom to see "User-Defined" settings. If there are no "User-Defined" settings, the project is set up to use [[#Convince Apple-Installed clang to Compile for OpenMP|Apple's compiler]]. If there are "User-Defined" settings, the <tt>CC</tt> setting specifies the <tt>clang-mp</tt> version being used. The next to sections explain how to change project settings if you want to use a different compiler or change to a different <tt>clang-mp</tt> version.


sudo port selfupdate
==== Project Setting for Apple's Compiler ====
sudo port install gcc49


The first command above is only needed to make sure your MacPorts is up to date. The second step will install the compiler at
Compiling using [[#Convince Apple-Installed clang to Compile for OpenMP|Apple's compiler]] in XCode requires these specific project build settings. Change as needed by editing the build settings for the <tt>NairnMPM</tt> and <tt>NairnFEA</tt> targets (both need to be edited):


/opt/local/bin/g++-mp-4.9
<ul>
<li><b>Compiler for C/C++/Objective-C</b>: select the default compiler.</li>
<li><b>Other Linker Flags</b>: add <tt>-lomp</tt> (replacing <tt>-fopenmp</tt> if there).</li>
<li><b>Runpath Search Paths</b>: remove <tt>/opt/local/libexec/llvm-17/lib</tt> (or with other clang version number) if it is there.</li>
<li><b>Header Search Paths</b>: remove <tt>/opt/local/include/libomp</tt> if it is there</li>
<li><b>Other C Flags</b>: add <tt>-Xclang -fopenmp</tt> together as a single flag (replacing <tt>-fopenmp</tt> if there).</li>
<li><b>Other C++ Flags</b>: add <tt>$(OTHER_CFLAGS)</tt> to have same flags used for C++ compile as well.</li>
<li>Scroll to bottom of build settings to see the "User-Defined" settings. If <b>CC</b>, <b>CLANG_CXX_LIBRARY</b>, and <b>LDPLUSPLUS</b> are there, delete them be selecting and using the delete key.</li>
</ul>


You can change to different version number if needed or when available. If you want to remove an older version of GCC after updating, such as removing GCC 4.8, you should be able to use
==== Project Settings for <tt>clang-mp</tt> ====


sudo port uninstall --follow-dependents gcc48
Compiling using a [[#Install clang-mp Compiler|<tt>clang-mp</tt> compiler]] in XCode requires these specific project build settings. Change as needed by editing the build settings for the <tt>NairnMPM</tt> and <tt>NairnFEA</tt> targets (both need to be edited):
 
To use GCC in XCode, you will need [[#Using GCC 4.8 or Newer in XCode|plug in and some specific build settings]]. For [[#MacOS X Command Line Compiling|command line compiling]], you may need to update the <tt>makefile</tt> to use you new compiler (see comments in <tt>makefile</tt> for details). Alternatively, you can use the <tt>CC</tt> option to specify path to any compiler, such as:
 
make SYSTEM=mac CC=/opt/local/bin/g++-mp-4.9
 
=== Using GCC 4.9 or Newer in XCode ===
 
Using GCC requires a hack. Fortunately, a friend of this project (Hammd Mazhar) has provided a solution. The process is explained on this [http://hamelot.co.uk/programming/add-custom-compiler-to-xcode/ blog] (for XCode 4.x) with an [http://hamelot.co.uk/programming/add-gcc-compiler-to-xcode-6/ update for Xcode 6.x and newer] (so far). <font color="red"><b>Warning</b></font>: Although the hack still allows one to compile in XCode 8.x, it no long appears possible to link the code using GCC. One alternative is to switch to [[#Install Clang-mp|switch to using <tt>clang-mp</tt>]].
 
In brief, the goal is to install a custom plug in. The shortest approach is download a plug in and install it:


<ul>
<li><b>Compiler for C/C++/Objective-C</b>: select the default compiler.</li>
<li><b>Other Linker Flags</b>: add <tt>-fopenmp</tt> (replacing <tt>-lomp</tt> if there).</li>
<li><b>Runpath Search Paths</b>: add <tt>/opt/local/libexec/llvm-17/lib</tt> (using clang version number you installed). This setting was not necessary until <tt>clang-mp-14</tt> (or maybe it was caused by MacOS Sonoma). Without this setting the executable fails to load the standard <tt>C++</tt> libraries. If you compile and run without this setting, you will get an error saying library not found. If you get that error, make sure this setting is correct.</li>
<li><b>Other C Flags</b>: add <tt>-fopenmp</tt> (replacing <tt>-Xclang -fopenmp</tt> if there).</li>
<li><b>Other C++ Flags</b>: add <tt>$(OTHER_CFLAGS)</tt> to have same flags used for C++ compile as well.</li>
<li><b>Header Search Paths</b>: add <tt>/opt/local/include/libomp</tt>
<li>Scroll to bottom of build settings to see the "User-Defined" settings. If <b>CC</b>, <b>CLANG_CXX_LIBRARY</b>, and <b>LDPLUSPLUS</b> are already there, make sure the <tt>clang-mp</tt> version numbers in <b>CC</b> and <b>LDPLUSPLUS</b> corresponds to one you have [[#Install clang-mp Compiler|installed]] or edit them to match your installed version. If any "User-Defined" settings are missible, add them with:
<ol>
<ol>
<li>For GCC 4.8 or GCC 4.9 installed as explained [[#Install GCC 4.8 or Newer|above]], download this [http://people.oregonstate.edu/~nairnj/FEAMPM/startup/GCCPlugin.zip GCC 4.8 XCode plug in] or this [http://people.oregonstate.edu/~nairnj/FEAMPM/startup/GCC49Plugin.zip GCC 4.9 XCode plug in]. For GCC newer then 4.9, you might find  a plug in in the repository [https://github.com/hmazhar/xcode-gcc xcode-gcc].</li>
<li>Use menu command Editor&rarr;Add Build Setting&rarr;Add User-Defined Setting and add setting for <b>CC</b> with value <b>/opt/local/bin/clang-mp-11</b> (using clang version number you installed).</li>
<li>For XCode 6.x through 8.x (and probably 5.x), the plug in must be installed in the application at:
<li>Use menu command Editor&rarr;Add Build Setting&rarr;Add User-Defined Setting and add setting for <b>LDPLUSPLUS</b> with value <b>/opt/local/bin/clang++-mp-11</b> (using clang version number you installed).</li>
<pre>/Applications/Xcode.app/Contents/Plugins/Xcode3Core.ideplugin/Contents\
<li>Use menu command Editor&rarr;Add Build Setting&rarr;Add User-Defined Setting and add setting for <b>CLANG_CXX_LIBRARY</b> with value <b>libc++</b>.</li>
/SharedSupport/Developer/Library/Xcode/Plug-ins/
</ol>
</pre>
</li>
<li>For Xcode 4.x it can alternatively be installed at
<pre>/Library/Application Support/Developer/Shared/Xcode/Plug-ins/
</pre></li>
<li>In your XCode project under "Other Warning Flags" remove the <tt>-Wmost</tt> option (if needed).
<li>You should now be able to select GCC 4.8 or GCC 4.9 from the compiler pop-up menu and compile code using that compiler.
</li>
</li>
<li>Note that this approach fails in El Capitan with XCode 7.x and fails when trying to link to the [[#Compiling Xerces on MacOS X|xerces library installed in its default location]]. One way to compile in El Capitan with XCode 7.x is:
<ul>
<li>Install GCC 4.9</li>
<li>Go to <tt>/opt/local/lib/gcc49</tt> and create a symbolic link to the xerces library using:
<pre>ln -s /usr/local/lib/libxerces-c-3.1.dylib libxerces-c-3.1.dylib
</pre>
Note that the symbolic link is to a versioned xerces file and not the standard  <tt>libxerces-c.dylib</tt> because linking to this link did not work. Because the xerces version is hard coded into the name, you will need to update when xerces is changed or alter the "3.1" if using a different version now. Although it should be possible to set Library Search Paths instead of creating this link, I could not get that approach to work, while the link did work.
</li>
<li>Make sure to select compiler GCC 4.9 in XCode</li>
<li>Add <tt>-fopenmp</tt> to "Other C++ Flags" to make sure it is compiled with OpenMP</li>
<li>Add <tt>-lgomp</tt> to "Other Linker Flags" to link to OpenMP library.</li>
</ul>
</ul>
</ol>


The second approach is to build the plug in yourself (this approach can also be used to customize the plug ins downloaded above). The details (from Hammad's [http://hamelot.co.uk/programming/add-custom-compiler-to-xcode/ blog 1] and [http://hamelot.co.uk/programming/add-gcc-compiler-to-xcode-6/ blog 2]) are as follows (and for GCC 4.9 or newer, replace all uses of 4.8 with the desired version):
=== Possible Changes Needed After Updating MacOS or XCode ===


<ol>
Unfortunately, Apple does not do a very good job of preserving your prior settings after a major upgrade and you will often need a complete reinstall of anything installed using MacPorts. Some things that might be needed with each MacOS upgrade are:
<li>For XCode 4.x, copy a current compiler plug in into to a <tt>/Library</tt> level plug in folder:
<ul>
<li>Go to XCode's plug-ins folder, which is in the application package:
<pre>cd /Applications/Xcode.app/Contents/PlugIns/Xcode3Core.ideplugin/Contents\
/SharedSupport/Developer/Library/Xcode/Plug-ins
</pre></li>
<li>Create a copy of "GCC 4.2.xcplugin", put it in the Xcode plugin folder (create that folder if needed), and go to its contents:
<pre>sudo mkdir -p "/Library/Application Support/Developer/Shared/Xcode/Plug-ins/"
sudo cp -r "GCC 4.2.xcplugin" "/Library/Application Support/Developer/Shared\
/Xcode/Plug-ins/GCC 4.8.xcplugin"
cd "/Library/Application Support/Developer/Shared/Xcode/Plug-ins/GCC 4.8.xcplugin/Contents"
</pre></li>
</ul></li>
<li>For XCode 5.x and newer you have to either download the plug in (see above) or copy the one mentioned in the previous step from a copy of XCode 4.x. Once copied, in must be installed in the new XCode app instead of the folder in the previous step. The correct folder is
<pre>/Applications/Xcode.app/Contents/PlugIns/Xcode3Core.ideplugin/Contents\
/SharedSupport/Developer/Library/Xcode/Plug-ins
</pre>
Once copied to that folder, navigate to the contents of the plug in:
<pre>cd /Applications/Xcode.app/Contents/PlugIns/Xcode3Core.ideplugin/Contents\
/SharedSupport/Developer/Library/Xcode/Plug-insGCC 4.8.xcplugin/Contents
</pre></li>
<li>Using Terminal app inside the plug in, convert the binary <tt>plist</tt> into text <tt>xml</tt> and then open for editing (done with <tt>vi</tt> here, but could use another tool):
<pre>sudo plutil -convert xml1 Info.plist
sudo vi Info.plist
</pre></li>
<li>Make the following changes:
<pre>"com.apple.xcode.compilers.gcc.42" -> "com.apple.xcode.compilers.gcc.48"
"GCC 4.2 Compiler Xcode Plug-in" -> "GCC 4.8 Compiler Xcode Plug-in"
</pre></li>
<li>Save and convert <tt>Info.plist</tt> back to binary:
<pre>sudo plutil -convert binary1 Info.plist
</pre></li>
<li>In the "Resources" folder rename two files:
<pre>cd Resources/
sudo mv GCC\ 4.2.xcspec GCC\ 4.8.xcspec
cd English.lproj/
sudo mv GCC\ 4.2.strings GCC\ 4.8.strings
</pre></li>
<li>Open the "GCC 4.8.xcspec" for editing (''e.g.'', using <tt>sudo vi</tt>) and make the changes:
<pre>Identifier = "com.apple.compilers.gcc.4_8";
Name = "GCC 4.8";
Description = "GNU C/C++ Compiler 4.8";
Version = "4.8";
ExecPath = "gcc-mp-4.8";
ShowInCompilerSelectionPopup = YES;
IsNoLongerSupported = NO;
</pre></li>
<li>Down further in that file, make the following changes:
<pre>under Name = "GCC_ENABLE_PASCAL_STRINGS"; set DefaultValue = NO;
under Name = "GCC_CW_ASM_SYNTAX"; set DefaultValue = NO;
</pre></li>
<li>Make [http://hamelot.co.uk/programming/add-gcc-compiler-to-xcode-6/ other customizations], if desired.</li>
<li>In your Xcode project under "Other Warning Flags" remove the <tt>-Wmost</tt> option (if needed).
</li>
<li>You should now be able to select GCC 4.8 from the compiler pop-up menu and compile code using that compiler.
</li>
</ol>


=== XCode Build Settings ===
# Get the latest version of XCode (from developer web site or app store).
 
# Reinstall command line tools in Terminal app by entering <tt>xcode-select --install</tt>
Compiling using a <tt>GCC</tt> compiler in Xcode requires some specific project build settings:
# Agree to XCode license in Terminal app using <tt>sudo xcodebuild -license</tt> (and provide your admin password).
 
# If you are using [[#Convince Apple-Installed clang to Compile for OpenMP|Apple's compiler]] you may need to update the <tt>libomp.dylib</tt> library for your new version of XCode.
<ol>
# If you are using MacPorts, you may need to uninstall it, install MacPorts for new MacOS, and then reinstall all previously installed ports. The process is explained in [https://trac.macports.org/wiki/Migration Migrating a MacPorts Installation]. Sometimes a "clean" install is best, especially after a major MacOS update. See [https://trac.macports.org/wiki/previous links] for details, but here is a brief outline of the process:
<li>Using <tt>GCC</tt> - these settings only work in XCode 7 or older
## [https://guide.macports.org/chunked/installing.macports.uninstalling.html Uninstall all of MacPorts]
<ul>
## [https://guide.macports.org/chunked/installing.macports.html Install MacPorts for the new MacOS]
<li><b>Compiler for C/C++/Objective-C</b>: select<tt>GCC 4.9</tt> (or version installed), which is available when [[#Using GCC 4.8 or Newer in XCode|plug in is properly installed]].</li>
## Renstall latest versions of all ports needed for this project ([[#Compiling for OpenMP|clang-mp]] and maybe [[OSParticulas#Installing svn in MacOS|subversion too]]). If you use different ports for other projects, you might want to use one of the advanced approaches on the [http://www.macports.org MacPorts] web site when migrating to new a new MacOS instead.
<li><b>Other C Flags</b>: add <tt>-fopenmp</tt>.</li>
# To compile in XCode, you may need more changes (but hard to predict what Apple will require next).
<li><b>Other C++ Flags</b>: add <tt>$(OTHER_CFLAGS)</tt> to have same flags used for C++ compile as well.</li>
<li><b>Other Linker Flags</b>: add <tt>-lgomp</tt>.</li>
<li><b>Other Warning Flags</b>: remove the <tt>-Wmost</tt> option</li>
</ul>
</li>
</ol>

Latest revision as of 13:37, 9 April 2024

This page explains several methods to compile code engines using MacOS X.

Using XCode

Most development of NairnMPM and NairnFEA is done in MacOS X and thus compiling on MacOS X should be easy. The preferred method is to use XCode (but first you must install XCode, install command line tools, and pick a compiler). A complete XCode project is located at

nairn-mpm-fea/Common/Projects/NairnMPM.xcodeproj

This project is called NairnMPM, but it includes all MPM and FEA source code in two targets named NairnMPM and NairnFEA (as well as targets for ExtractMPM and CompareGlobal). Once xerces is installed, open the NairnMPM.xcodeproj to compile the code by:

  1. Select the target to compile - either NairnMPM or NairnFEA (or ExtractMPM or CompareGlobal)
  2. Choose build and they will compile and be saved in XCode's derived data folder.

If a compile error occurs, you might need to change project settings to match you method for compiling for OpenMP.

If a linking error occurs, you might have conflicting architectures between the XCode settings and the xerces library you installed or be missing some important C++ libraries. To fix xerces problems edit both the project and target settings and under the "Architectures" section, set the "Architectures" option to match the architecture you used when installing xerces or recompile xerces if needed. C++ library problems are sometimes caused by MacOS updates and might need some new project settings.

Compiling Xerces on MacOS X

Before you can compile and run the project, however, you will need an installed version of the xerces library and a copy of the xerces header files. These compiling instructions assume they are installed in the default locations for MacOS X or at:

/usr/local/lib/libxerces-c.dylib

for the library and at

/usr/local/include

for the header files. These can be changed if needed by editing the project and target settings. The nairn-mpm-fea project needs access to xerces 3.0 or newer. This libraries can be obtained by downloading the xerces source code from the Apache Software Foundation web site and then building and installing xerces with the following steps:

  1. First install XCode and command line tools.
  2. Open Terminal app and navigate to the xerces folder expanded from the downloaded file.
  3. Configure the code with the command:
    ./configure CFLAGS="-arch x86_64 -arch arm64" CXXFLAGS="-arch x86_64 -arch arm64"
    where the provided arch options (or architectures) are for Intel chips (x86_64) or the new Silcon M1 chips (arm64); you can pick one for your computer or include both to have a "fat" binary with both architectures. To determine your chips, choose "About This Mac" from the Apple menu.
  4. When the configuration is done, use the following commands:
    cd src
    make
    sudo make install
    These commands make the library (but not the unneeded xerces examples). The final install command after make is done, installs both the library and the header files at the default locations listed above. It requires sudo for you to provide your administrator password needed to authenticate installation and the default location (which is /usr/local).
  5. When working with nairn-mpm-fea on a new Mac, the xerces installation only needs to be done once. The only reason to repeat it is when a new xerces version is available and/or the project requires a new version for compatibility.

MacOS X Command Line Compiling

It also possible to compile on MacOS X using a command line approach. After installing XCode, command line tools, and a compiler and after installing xerces and its header files, you can compile NairnMPM using:

cd nairn-mpm-fea/NairnMPM/build
make -j 8 SYSTEM=mac (or SYSTEM=mac-clang)

and compile NairnFEA using:

cd nairn-mpm-fea/NairnFEA/build
make -j 8 SYSTEM=mac (or SYSTEM=mac-clang)

The -j 8 option compiles in parallel where the number is number of available cores on your computer. The SYSTEM setting tells the make process what compiler to use. Use SYSTEM=mac if you have modified Apple's compiler or SYSTEM=mac-clang if you installed a separate clang-mp.

The above command will compile the source cade and the executables will be installed in nairn-mpm-fea/NairnMPM/input or nairn-mpm-fea/NairnFEA/input, respectively. You can use an additional make install command to copy each compiled executable to your ~/bin folder if desired.

If the command-line compile does not work, the most likely explanations are:

  1. A problem with your compiler settings. Two options are to to edit the makefile to use your chosen compiler or to pass alternate parameters to the make command.
  2. A problem with the xerces installation. You either have to install it as specified above or edit the makefile to recognize your custom installation. The process is documented in the makefile and involves editing the xercesSo and headersPath variables for your different settings.

Compiling Old Version of the Code

You can change settings used in the make process without editing the makefile by passing alternate parameters to the make command. One reason to use this approach is to compile old versions of the code. If old versions are checked out, the XCode project may not work (because of changes in XCode versions, build settings, or compiler you have installed). When this occurs, you can use command-line compile methods instead. For example, if working with current code on on Mac and the clang-mp compiler, you may be able to compile old versions using the build folders and the command

make CC=/opt/local/bin/clang-mp-11 CCLINK=/opt/local/bin/clang++-mp-11 \
        xercesSo=/usr/local/lib/libxerces-c.dylib \
        CFLAGS="-c -O3 -fopenmp -std=c++11 -I/opt/local/include/libomp" \
        LFLAGS="-fopenmp -lc++"

where CC and CCLINK refers to a clang-mp compiler and linker you have installed (old versions of this project do not use CCLINK). The other parameters might also need changing. You can open the project's makefile to see its default options for these variables and then change as needed.

When working with current version of the project, a better approach than using command-line arguments is to edit the makefile and pick your own default settings. Support for using a modifed Apple compiler should be built in already using the SYSTEM=mac setting. To add support for a different version of clang-mp, just follow examples used for others versions in the current makefile.

Installing XCode

The simplest way to compiling on Mac requires that you install XCode. You can get XCode from the Mac App Store or from the Apple Developer site. It is a large install. Once installed, you then need to install command line tools. Unfortunately, the compilers provided in Apple's "Command Line Tools" do not support OpenMP used to make these code engines parallel. To compile for parallel calculations (which is important), see the two options below to compile for OpenMP. Whenever you update your MacOS or XCode, you may need to repeat some installation steps before you can return to compiling the code engines.

The following sections have more details on installations needed for compiling the code engines.

Installing Command Line Tools

Apple used to install command line tools by default, but stopped doing that a long time ago. You now have to manually install the tools before you can compile code. These steps should work:

  1. The simplest process is to open the Terminal app and enter:
    xcode-select --install
    This install may or may not require XCode to already be installed. If it fails try installing XCode first. If it stall fails, refer to MacPorts documentation or try installing HomeBrew (which I think handles command-line too installation as well.
  2. You may need to agree to XCode license by using Terminal app to enter
    sudo xcodebuild -license
  3. Most tools should now be available. You should be able to make and install the xerces library.
  4. To compile the code engines for parallel code, you will additionally need to install a compiler that works with OpenMP (none is provided in the command line tools).

With each new system, Apple creates road blocks to keep you from using you computer for interesting programming. For now, by using a few tricks it is still possible to get around their road blocks. Hopefully these tricks will continue to work in the future.

One issue in the XCode project is that it might not find your xerces library, even if it is in the standard location. If it is not found, the library may be in red in the "External Frameworks and Libraries" folder (or may not). A potential solution is to delete the reference to the library and then add it back. The problem is that you cannot navigate to the /usr/local/lib folder anymore (another new Apple "feature"). Here is a trick to get there:

  1. In the Finder, use the "Go to Folder..." menu command and enter "/usr/local".
  2. After it opens, choose the "Add To Sidebar" menu command. This folder will now appear in all file selection boxes.
  3. Go back to XCode and use command to add files for adding the xerces library.
  4. Go through your new "local" folder and select the xerces library file in /usr/local/lib.

Compiling for OpenMP

Although Apple uses clang for compiling in XCode, for reasons that are hard to justify, they opted to disable compiling with OpenMP (i.e., they disabled the -fopenmp compiler option). They further did not include libomp.dylib, which is needed to run software written for OpenMP. Because doing parallel computations is essential for large calculations, some method to use OpenMP is needed. Two possible solutions are given in this section.

Convince Apple-Installed clang to Compile for OpenMP

The approach with minimal installation is to override Apple's disabling of the -fopenmp option to compile for OpenMP anyway. This approach uses the following steps:

  1. The only extra installation required is a signed copy of the libomp.dylib. Fortunately. a member of the R project has made this installation easy. First check your version of XCode. Next, look up file name you need in the following table:
    XCode Version File name to download
    15.0+ openmp-16.0.4-darwin20-Release.tar.gz
    14.3.x openmp-15.0.7-darwin20-Release.tar.gz
    14.0 to 14.2 openmp-14.0.6-darwin20-Release.tar.gz
    13.3 to 13.4.1 openmp-13.0.0-darwin21-Release.tar.gz
    13.0 to 13.2.1 openmp-12.0.1-darwin20-Release.tar.gz
    If you have a newer version of XCode, you can check the R project posting for a new file name. If you have an older version of XCode, now is a good time to update your Mac.
  2. Once you have the file name, download it from r-project.org in the Terminal app using:
    curl -O https://mac.r-project.org/openmp/openmp-15.0.7-darwin20-Release.tar.gz

    replacing the file name with the one you need for your version of XCode. If you visit the R project posting, do not download with the links in that web site because Apple will quarantine all files rendering the library unusable without more work.

  3. Once downloaded, extract and install libomp files with
    sudo tar fvxz openmp-15.0.7-darwin20-Release.tar.gz -C /
    being sure to replace the file name in this command with the one downloaded in the previous step.
  4. The -C / option in the previous step extracts the files directly to your root /usr/local folder rather a folder next to the downloaded file. When done, the following files should be installed:
    /usr/local/lib/libomp.dylib
    /usr/local/include/ompt.h
    /usr/local/include/omp.h
    /usr/local/include/omp-tools.h

Once these steps are done, you are ready to compile in XCode or compile on a command line (using the SYSTEM=mac option). If an XCode compile has problems, you may have adjust some of the XCode project settings for this compiler option.

According to the R project posting, this approach works now, but " may break at any time." Furthermore, each time you upgrade XCode, you may need to install a signed and updated version of libomp.dylib. If for some reason the method stops working or an updated libomp.dylib cannot be found, you can switch to an alternate method that involves installing a separate compiler (and is explain in the next section).

Install clang-mp Compiler

An alternate approach to compiling parallel code using OpenMP is to install a separate clang-mp compiler. One way to install this compiler is to use MacPorts as follows:

  1. If not already installed, install MacPorts. Be aware that it might need to be reinstalled after updating your MacOS.
  2. Open the Terminal app and enter
    sudo port selfupdate
    sudo port install clang-11 ld64 +ld64_xcode

    The first command above is only needed to make sure your MacPorts is up to date (you might need to reinstall instead of selfupdate after updating your MacOS). The second step installs the compiler version requested (change "11" to version you want)

  3. To verify it installed, check in Terminal app:
    which clang-mp-11
    /opt/local/bin/clang-mp-11

    If the which command did not return the installed location, an error occurred during install.

The above install usually works. Some issues have occurred in MacOS Sonoma and new issues might happen in future MacOS versions. Here are some known issues:

  1. A MacPorts install command in MacOS Sonoma reports that clang-mp versions 11, 12, and 13 are "known to fail." It appears these version can still be installed, but the sudo port install may not finish. If it exits without a binary, repeat the install command until it (hopefully) finishes. Each repeat picks up where it left off. If it hangs up while installing, force exit with Control-C and repeat again. If it eventually finishes, the install seems to work. I needed three repeats for one successful install of clang-mp-11.
  2. Installing clang-mp versions 14 or newer in MacOS Sonoma does not say "known to fail", but the first install command may exit with an unexpected "permission denied" error. Repeating the install command then picks up where it left off and finishes the install.
  3. Although these versions get installed, you may need to verify the XCode project settings match both your installed version of clang-mp and your choice to use a separate compiler. A new issue occurs in MacOS Sonoma, but that is corrected with a new runpath project setting. Future MacOS's might cause more problems and some might be fixable with new project settings.

You can install multiple versions of clang-mp, which can be useful if working with multiple checkouts of this project. More likely, you only need one version. If you settle on one version, you can remove older ones with

sudo port uninstall --follow-dependents clang-xx

where xx is the version you want to uninstall.

XCode Build Settings

The downloaded XCode project will have project settings for compiling with Apple's compiler or with an installed clang-mp compiler. To find out which one, open the project in XCode, find the "Build Settings" for the NairnMPM or NairnFEA target, and scroll the to bottom to see "User-Defined" settings. If there are no "User-Defined" settings, the project is set up to use Apple's compiler. If there are "User-Defined" settings, the CC setting specifies the clang-mp version being used. The next to sections explain how to change project settings if you want to use a different compiler or change to a different clang-mp version.

Project Setting for Apple's Compiler

Compiling using Apple's compiler in XCode requires these specific project build settings. Change as needed by editing the build settings for the NairnMPM and NairnFEA targets (both need to be edited):

  • Compiler for C/C++/Objective-C: select the default compiler.
  • Other Linker Flags: add -lomp (replacing -fopenmp if there).
  • Runpath Search Paths: remove /opt/local/libexec/llvm-17/lib (or with other clang version number) if it is there.
  • Header Search Paths: remove /opt/local/include/libomp if it is there
  • Other C Flags: add -Xclang -fopenmp together as a single flag (replacing -fopenmp if there).
  • Other C++ Flags: add $(OTHER_CFLAGS) to have same flags used for C++ compile as well.
  • Scroll to bottom of build settings to see the "User-Defined" settings. If CC, CLANG_CXX_LIBRARY, and LDPLUSPLUS are there, delete them be selecting and using the delete key.

Project Settings for clang-mp

Compiling using a clang-mp compiler in XCode requires these specific project build settings. Change as needed by editing the build settings for the NairnMPM and NairnFEA targets (both need to be edited):

  • Compiler for C/C++/Objective-C: select the default compiler.
  • Other Linker Flags: add -fopenmp (replacing -lomp if there).
  • Runpath Search Paths: add /opt/local/libexec/llvm-17/lib (using clang version number you installed). This setting was not necessary until clang-mp-14 (or maybe it was caused by MacOS Sonoma). Without this setting the executable fails to load the standard C++ libraries. If you compile and run without this setting, you will get an error saying library not found. If you get that error, make sure this setting is correct.
  • Other C Flags: add -fopenmp (replacing -Xclang -fopenmp if there).
  • Other C++ Flags: add $(OTHER_CFLAGS) to have same flags used for C++ compile as well.
  • Header Search Paths: add /opt/local/include/libomp
  • Scroll to bottom of build settings to see the "User-Defined" settings. If CC, CLANG_CXX_LIBRARY, and LDPLUSPLUS are already there, make sure the clang-mp version numbers in CC and LDPLUSPLUS corresponds to one you have installed or edit them to match your installed version. If any "User-Defined" settings are missible, add them with:
    1. Use menu command Editor→Add Build Setting→Add User-Defined Setting and add setting for CC with value /opt/local/bin/clang-mp-11 (using clang version number you installed).
    2. Use menu command Editor→Add Build Setting→Add User-Defined Setting and add setting for LDPLUSPLUS with value /opt/local/bin/clang++-mp-11 (using clang version number you installed).
    3. Use menu command Editor→Add Build Setting→Add User-Defined Setting and add setting for CLANG_CXX_LIBRARY with value libc++.

Possible Changes Needed After Updating MacOS or XCode

Unfortunately, Apple does not do a very good job of preserving your prior settings after a major upgrade and you will often need a complete reinstall of anything installed using MacPorts. Some things that might be needed with each MacOS upgrade are:

  1. Get the latest version of XCode (from developer web site or app store).
  2. Reinstall command line tools in Terminal app by entering xcode-select --install
  3. Agree to XCode license in Terminal app using sudo xcodebuild -license (and provide your admin password).
  4. If you are using Apple's compiler you may need to update the libomp.dylib library for your new version of XCode.
  5. If you are using MacPorts, you may need to uninstall it, install MacPorts for new MacOS, and then reinstall all previously installed ports. The process is explained in Migrating a MacPorts Installation. Sometimes a "clean" install is best, especially after a major MacOS update. See links for details, but here is a brief outline of the process:
    1. Uninstall all of MacPorts
    2. Install MacPorts for the new MacOS
    3. Renstall latest versions of all ports needed for this project (clang-mp and maybe subversion too). If you use different ports for other projects, you might want to use one of the advanced approaches on the MacPorts web site when migrating to new a new MacOS instead.
  6. To compile in XCode, you may need more changes (but hard to predict what Apple will require next).