Thursday, March 3, 2011

Targeting both 32bit and 64bit with Visual Studio in same solution/project

Greetings.

I have a little dilemma on how to set up my visual studio builds for multi-targeting.

Background: c# .NET v2.0 with p/invoking into 3rd party 32 bit DLL's, SQL compact v3.5 SP1, with a Setup project. Right now, the platform target is set to x86 so it can be run on Windows x64.

The 3rd party company has just released 64 bit versions of their DLL's and I want to build a dedicated 64bit program.

This raises some questions which I haven't got the answers to yet. I want to have the exact same code base. I must build with references to either the 32bit set of DLL's or 64bit DLL's. (Both 3rd party and SQL Server Compact)

Can this be solved with 2 new sets of configurations (Debug64 and Release64) ?

Must I create 2 separate setup projects(std. visual studio projects, no Wix or any other utility), or can this be solved within the same .msi?

Any ideas and/or recommendations would be welcomed.

/M

From stackoverflow
  • Not sure of the total answer to your question - but thought I would point out a comment in the Additional Information section of the SQL Compact 3.5 SP1 download page seeing you are looking at x64 - hope it helps.

    Due to changes in SQL Server Compact SP1 and additional 64-bit version support, centrally installed and mixed mode environments of 32-bit version of SQL Server Compact 3.5 and 64-bit version of SQL Server Compact 3.5 SP1 can create what appear to be intermittent problems. To minimize the potential for conflicts, and to enable platform neutral deployment of managed client applications, centrally installing the 64-bit version of SQL Server Compact 3.5 SP1 using the Windows Installer (MSI) file also requires installing the 32-bit version of SQL Server Compact 3.5 SP1 MSI file. For applications that only require native 64-bit, private deployment of the 64-bit version of SQL Server Compact 3.5 SP1 can be utilized.

    I read this as "include the 32bit SQLCE files as well as the 64bit files" if distributing for 64bit clients.

    Makes life interesting I guess.. must say that I love the "what appears to be intermittent problems" line... sounds a bit like "you are imagining things, but just in case, do this..."

  • Hi,

    Regarding your last question. Most likely you cant solve this inside a single MSI. If you are using registry/system folders or anything related, the MSI itself must be aware of this and you must prepare a 64bit MSI to properly install on 32 bit machine.

    There is a possibility that you can make you product installed as a 32 it application and still be able to make it run as 64 bit one, but i think that may be somewhat hard to achieve.

    that being said i think you should be able to keep a single code base for everything. In my current work place we have managed to do so. (but it did took some juggling to make everything play together)

    Hope this helps. Heres a link to some info related to 32/64 bit issues: http://blog.typemock.com/2008/07/registry-on-windows-64-bit-double-your.html

  • You should use the Configuration Manager to handle compiling the two different versions of your P/Invoke calls. However, you will not be able to do what you'd like with a single MSI.

    From How to: Create a Windows Installer for a 64-bit Platform (emphasis added):

    64-bit MSI packages can support both 32- and 64-bit DLLs or EXEs to perform custom actions; however, 32-bit MSI packages cannot support 64-bit DLLs or EXEs to perform custom actions.

    ...

    To distribute a .NET Framework application both to 32- and 64-bit platforms, build two MSI packages, one targeted at a 32-bit and the other a 64-bit computer. The user can install the 32-bit setup project package and it will most likely run as a 64-bit application, only the 64-bit setup project will install to the "Program Files64" directory.

  • Yes, you can target both x86 and x64 with the same code base in the same project. In general, things will Just Work if you create the right solution configurations in VS.NET (although P/Invoke to entirely unmanaged DLLs will most likely require some conditional code): the items that I found to require special attention are:

    • References to outside managed assemblies with the same name but their own specific bitness (this also applies to COM interop assemblies)
    • The MSI package (which, as has already been noted, will need to target either x86 or x64)
    • Any custom .NET Installer Class-based actions in your MSI package

    The assembly reference issue can't be solved entirely within VS.NET, as it will only allow you to add a reference with a given name to a project once. To work around this, edit your project file manually. After adding a reference to, say, the x86 version of an assembly, your project file will contain something like:

    <Reference Include="Filename, ..., processorArchitecture=x86">
      <HintPath>C:\path\to\x86\DLL</HintPath>
    </Reference>
    

    Wrap that Reference tag inside an ItemGroup tag indicating the solution configuration it applies to, e.g:

    <ItemGroup Condition=" '$(Configuration)|$(Platform)' == 'Debug|x86' ">
       <Reference ...>....</Reference>
    </ItemGroup>
    

    Then, copy and paste the entire ItemGroup tag, and edit it to contain the details of your 64-bit DLL, e.g.:

    <ItemGroup Condition=" '$(Configuration)|$(Platform)' == 'Debug|x64' ">
      <Reference Include="Filename, ..., processorArchitecture=AMD64">
         <HintPath>C:\path\to\x64\DLL</HintPath>
       </Reference>
    </ItemGroup>
    

    After reloading your project in VS.NET, the Assembly Reference dialog will be a bit confused by these changes, and you may encounter some warnings about assemblies with the wrong target processor, but all your builds will work just fine.

    Solving the MSI issue is up next, and unfortunately this will require a non-VS.NET tool: I prefer Caphyon's Advanced Installer for that purpose, as it pulls off the basic trick involved (create a common MSI, as well as 32-bit and 64-bit specific MSIs, and use an .EXE setup launcher to extract the right version and do the required fixups at runtime) very, very well.

    You can probably achieve the same results using other tools or the Windows Installer XML (WiX) toolset, but Advanced Installer makes things so easy (and is quite affordable at that) that I've never really looked at alternatives.

    One thing you may still require WiX for though, even when using Advanced Installer, is for your .NET Installer Class custom actions. Although it's trivial to specify certain actions that should only run on certain platforms (using the VersionNT64 and NOT VersionNT64 execution conditions, respectively), the built-in AI custom actions will be executed using the 32-bit Framework, even on 64-bit machines.

    This may be fixed in a future release, but for now (or when using a different tool to create your MSIs that has the same issue), you can use WiX 3.0's managed custom action support to create action DLLs with the proper bitness that will be executed using the corresponding Framework.

    Oliver Bock : In Visual Studio 2008, I found that s could not be nested. This solution worked fine once I made the new s below the group the rest of the s. I also had to change x86 to AnyCPU, which probably relates to the history of my particular project.
  • Let's say you have the DLLs build for both platforms, and they are in the following location:

    C:\whatever\x86\whatever.dll
    C:\whatever\x64\whatever.dll
    

    You simply need to edit your .csproj file from this:

    <HintPath>C:\whatever\x86\whatever.dll</HintPath>
    

    To this:

    <HintPath>C:\whatever\${Platform}\whatever.dll</HintPath>
    

    You should then be able to build your project targeting both platforms, and MSBuild will look in the correct directory for the chosen platform.

    John Sheehan : This would be brilliant if it worked, but it doesn't. At least not for me.
    Andreas : Isn't that supposed to be: C:\whatever\$(Platform)\whatever.dll
    Bart Janson : this worked miracles for me, thanks a bundle!
    Daok : This solution worked for me with VS2010.
    Oliver Bock : Worked OK on Visual Studio 2008 for me, but did not automatically copy the DLL to the build target directory, as a normal does. mdb's solution worked better for me.
  • If you use Custom Actions written in .NET as part of your MSI installer then you have another problem.

    The 'shim' that runs these custom actions is always 32bit then your custom action will run 32bit as well, despite what target you specify.

    More info & some ninja moves to get around (basically change the MSI to use the 64 bit version of this shim)

    Building an MSI in Visual Studio 2005/2008 to work on a SharePoint 64

    64-bit Managed Custom Actions with Visual Studio

  • Hi,

    I have installed Visual studio 2008 on 64-bit windows server 2008 r2, But everything got installed under Program files(x86). which means 32-bit,

    Please throw some light on 64-bit installation of vS2008.

    Thanks in advance Krishna

    Magnus Johansson : The Visual Studio 2008 IDE is 32bit only, so what you are seeing is normal, but it can target x64 builds as well.
    jeffamaphone : This is not an answer, but another question in and of itself.
  • VS is a 32 bit app, it will most likely installs under the x86 part of your programs :)

0 comments:

Post a Comment