README.MS-VisualC.md:
It is no longer necessary to set up file copying in the post-installation
step to have a version of the Mercury compiler using MSVC that works on
the Windows command prompt. The configuration script now chooses to use
internal file copying by default when using MSVC.
8.8 KiB
Mercury with Microsoft Visual C++
Mercury has been ported to use the Microsoft Visual C++ compiler (MSVC). It requires at least version 12.0 (2013); older versions lack sufficient C99 support. Version 19.0 (2022) is recommended when using the 64-bit (x64) version of MSVC. Support for 64-bit code and MSVC has not been tested with older versions.
Setting up the build environment
A Unix-like environment is required for building and installing Mercury. Any one of Cygwin, MSYS or MSYS2 will suffice.
A Unix-like environment is not required in order to use Mercury once it has been installed.
To make MSVC and its supporting tools available under the Cygwin, MSYS or MSYS2 shells do the following:
-
Open the Visual Studio Command Prompt.
This can typically be found in
Start -> Visual Studio YYYY -> Visual Studio Toolsalthough the name and location vary between different versions and editions of Visual Studio. (YYYY is the year.) Note that this will give you the x86 version of MSVC.
The Visual Studio entry in the
Startmenu may also have items named similarly to:* x64 Native Tools Command Prompt for VS 2022 * x86 Native Tools Command Prompt for VS 2022These allow you to select between the x64 and x86 versions of MSVC toolchain.
Alternatively, you can start a
cmd.exesession and use the batch files supplied with Visual Studio (e.g.vcvarsall.bat,vcvars32.batorvcvars64.bat) to set up the environment for MSVC. -
Enter the following command to start the MSYS shell:
C:\> C:\MinGW\MSYS\1.0\msys.bator this one to start the MSYS2 shell with 32-bit MSVC:
C:\> C:\msys64\msys2_shell.cmd -use-full-path -mingw32or this one to start the MSYS2 shell with 64-bit MSVC:
C:\> C:\msys64\msys2_shell.cmd -use-full-path -mingw64or this one to start the Cygwin shell:
C:\> C:\cygwin64\Cygwin.bat
(We assume the default installation locations for MinGW/MSYS and Cygwin above.)
Users of MSYS2 should note that it is required that the configuration type
returned by autoconf match *mingw*; in particular it must not match
*-pc-msys. You can use the config.guess script to see what configuration
type autoconf detects.
To install the C# or Java grades, you will need a C# or Java compiler
respectively to be included in the Windows PATH.
(See the relevant README files for further details, e.g.
README.Java.md etc)
Configuration and Installation
The MSVC port of Mercury is compatible with the prebuilt C files contained in
the Mercury source distribution. The asm_fast* and reg* grades will not work
with MSVC (see below). When using the prebuilt C files the compiler will be
built in the hlc.gc.pregen grade.
Alternatively, if you have an existing Mercury installation that uses the MinGW or Cygwin GCC ports, or clang, then you can checkout the Mercury source from the git repository and use your existing installation to cross-compile the MSVC port.
In either case, to use MSVC as the C compiler with Mercury, invoke configure
as follows:
$ ./configure --with-cc=cl [--with-msvcrt] [<any other options>]
The --with-msvcrt flag causes executables built with this install of Mercury
to be linked with the MS Visual C runtime, instead of the standard libC
runtime.
On Cygwin, configure and mmake will do translation of Unix style paths, so
you may specify the installation directory using either a Unix- or
Windows-style path. On MSYS or MSYS2, you must use a full Windows-style path
with a drive letter, except that you must use / instead of \ as a directory
separator. For example, this is acceptable:
$ ./configure --prefix="c:/where/to/install/mercury"
but this is not:
$ ./configure --prefix="c:\where\to\install\mercury"
Once configure has successfully finished, then you should do
$ make
$ make install
as normal.
Limitations
The MSVC port currently has a number of limitations:
-
The
asm_fastandreggrades do not work with MSVC. Both use GNU extensions to C that MSVC does not provide. -
Time profiling does not (currently) work with MSVC. Time profiling grades (those whose name contains the
profgrade component) will not be installed.Note that memory profiling does work with MSVC. (Memory profiling grades are those whose name contains the
memprofgrade component.) -
Parallel grades (those whose name contains the
parcomponent) do not currently work with MSVC.In parallel grades the Mercury runtime currently requires the use of POSIX threads; it has not currently been ported to use Windows threads. (It might be possible to use the pthreads-win32 library with MSVC to provide POSIX threads but we have not tested that yet.)
-
Deep profiling (e.g. the
*.profdeep grades) does not (currently) work with MSVC. (In principle, it should work if the clock tick metric is disabled.) -
The deep profiling tool (
mdprof_cgi) does not currently work with MSVC. This is due to it containing a number of Unix dependencies. (Other tools that work with deep profiles should be fine.) -
When used directly from the Windows command prompt, mmake will not work. You should use
mmc --makeinstead. (mmakerequires a POSIX-like shell and GNUmake; it will however work with the Cygwin or MSYS shells.) We do not intend to supportmmakedirectly on Windows. -
Creation of shared libraries (DLLs) is not currently supported.
-
The
--c-debugoption currently has no effect with MSVC since enabling it breaks parallel builds and disables some C compiler optimizations.If you really want to enable support for C level debugging, then enable the commented out definition of
DEBUG_OPTSinscripts/mgnuc.in(in theclcase) and also enable the commented out definition ofCFLAGS_FOR_DEBUGinconfigure.ac(in themsvc*case). You will then need to regenerate theconfigurescript and rebuild the Mercury system. (See INSTALL.git for details of how to do this.)
Post-installation configuration
The above instructions create a Mercury installation that targets MSVC that
works from within the Cygwin, MSYS or MSYS2 shells. If you want to be able to
run the Mercury compiler directly from the Windows command prompt (e.g.
cmd.exe) then you need to manually edit some configuration files in the
installation. (In future releases, this will all hopefully be automated.)
All references to files in the following are within the Mercury installation directory
-
In the file
lib/mercury/conf/Mercury.config:-
Replace any Unix-style paths with their Windows-style equivalent.
-
Modify the values of the options
--host-env-typeand--target-env-typein the value of the variableDEFAULT_MCFLAGSso that their values are as follows:--host-env-type "windows" --target-env-type "windows"
-
-
In the file
lib/mercury/mdb/mdbrc-
The backslash character,
\, is used as an escape character in mdbrc files. You need to escape it if it occurs in any paths used in the argument of source commands, for examplesource c:\mercury-11.07\lib\mercury\mdb\mdb_docneeds to be replaced with:
source c:\\mercury-11.07\\lib\\mercury\\mdb\\mdb_doc -
Delete the aliases for the
openandgrepcommands. The scripts that implement these commands assume a Unix-style environment.
-
-
The
bindirectory contains batch files equivalent to themmc,mdbandmprofscripts. Note that the batch file equivalent to themmcscript is namedmercury.batin order to avoid clashing with the executable for the Microsoft Management Console. -
The following scripts do not currently have a Windows equivalent.
mprof_merge_runsmtc
-
The other shell scripts in the
bindirectory do not have (or need) Windows equivalents. (Most of them are part of the implementation ofmmakewhich is not supported on Windows systems.)
Installing on network drives (Cygwin only)
If you want to install on the machine, foo, in the directory \usr\local
you need to add --prefix //foo/usr/local as an option to configure.
Then ensure that the directory that you wish to install into is mounted somewhere, i.e.
$ mount //foo/usr/local /temp_mount_point
and then use that mount point as the location to install into
$ make install INSTALL_PREFIX=/temp_mount_point
This avoids the problem that mkdir -p //foo/usr/local/xxx fails.