Windows – Is it possible to “declare” a windows cmd-file (batch-file) as 32 bit

64-bitcommand lineenvironment-variableswindows 7windows-registry

Background:

We do quite some scripting and little helpers with batch files (xyz.cmd) at work. Windows 7 is only now starting to become widespread here, and obviously, we hit the problems associated with the different environment variables of 32bit vs. 64bit windows.

Specifically, if you run C:\Windows\system32\cmd.exe on a 64 bit Windows, you'll get:

...
ProgramFiles=C:\Program Files
ProgramFiles(x86)=C:\Program Files (x86)
ProgramW6432=C:\Program Files
...

whereas, if you start C:\Windows\SysWOW64\cmd.exe on a 64 bit Windows, you'll get:

...
ProgramFiles=C:\Program Files (x86)         <-- NOTE
ProgramFiles(x86)=C:\Program Files (x86)
ProgramW6432=C:\Program Files
...

It also so happens, that because this cmd.exe is a 32bit process, it also gets all the other SysWOW64 redirections "for free" — any invocation of regedit will go to the 32bit registry, etc.

This can be extremely useful if the batch file is supposed to do some tasks related to a 32bit application, e.g. path + registry stuff.

Question:

GIven that I have a class of batch files that I would wish to always run with the 32bit version of cmd.exe, is there an — easy! — way to force these batchfiles to always use the 32bit cmd.exe on 64bit versions of Windows and run normally on 32bit versions of Windows?

Obviously I can add a "header" to each such batch-file to change environment variables and regedit invocations and/or obviously I could just tell users to "run this batch file via the SysWOW64 cmd.exe", but neither of these solutions seems very attractive 🙂

Best Answer

I have three possible solutions for your problem:

1) Use a wrapper batch file

In a company I used to work for, we had a single wrapper batch file for everything we ran. We did this to simplify interaction with our users, but it would work similarly. If you wanted, you could create a wrapper for each batch file (in essence, creating two batch files), or create a single wrapper that allows you to select each batch file from a menu (as described http://http-server.carleton.ca/~dmcfet/menu.html). You could have each menu selection identify the path to the batch file to run, and then you could have the wrapper choose the correct cmd.exe to launch it with.

2) Use the batch file as a self-wrapper

For each batch file, you could include a header which determines what OS you are on (32 or 64-bit). If you are on a 64-bit OS, you would know that, by default, you are launching it using the 64-bit version of cmd.exe. You could have that file then launch the 32-bit version of cmd.exe with the same batch file, but you could also pass the batch file a flag which tells it to ignore the 64-bit check. It would then run under the correct 32-bit cmd.exe.

For example:

@ECHO Off

::Check if 64-bit check skip flag exists

IF %1 == /skipcheck (goto run)

::Check if OS is 64-bit

IF %processor_architecture%==AMD64 (<path_to_32-bit_cmd> /c "<path_to_batch_file>\<name_of_my_batch_file>" /skipcheck)
IF %processor_architecture%==AMD64 (goto end)

:run

echo Hello World!

:end

I haven't tested the above code, so I might have some quotes or parenthesis in the wrong spot, but this is the general idea.

3) Migrate to PowerShell

PowerShell is available for all Windows OSes XP and newer. PowerShell is designed by Microsoft to eventually replace the simple cmd.exe, and provides a rich language for doing nearly any task. Most batch programs are actually runnable in PowerShell, and at most may require minimal tweaking.

Related Question