September 14th, 2020

C11 and C17 Standard Support Arriving in MSVC

Elnar Dakeshov
Software Engineer 2

Please see our Visual Studio 2019 version 16.8 Preview 3 release notes for more of our latest features.

Update: Windows 10 SDK version 2104 has been released, which includes the changes needed for C11 and C17 as well as the conformant preprocessor. It can be downloaded here. To use this SDK, follow the instructions from step 3.

Our team is happy to announce that C11 and C17 are becoming supported language versions in the MSVC compiler toolset starting with Visual Studio 2019 version 16.8 Preview 3!

For many years Visual Studio has only supported C to the extent of it being required for C++. Things are about to change now that a conformant token-based preprocessor has been added to the compiler. With the advent of two new compiler switches, /std:c11 and /std:c17, we are officially supporting the latest ISO C language standards.

What’s in:

All the required features of C11 and C17 are supported. This meant adding the following functionalities:

  • _Pragma
  • restrict
  • _Noreturn and <stdnoreturn.h>
  • _Alignas, _Alignof and <stdalign.h>
  • _Generic and <tgmath.h> support
  • _Static_assert

IntelliSense works natively with these features as well, simply use a .c file extension for your source files or the /TC compiler switch to enable syntax highlighting for C code.

Currently IntelliSense highlighting is only available for the keywords and not the macros introduced by the standard headers. This will be fixed in a later release.

Since C17 is essentially just a bug fix release of ISO C, with many defect reports being adopted, our support for C11 already includes all the relevant defect reports. At present, there are no differences between the C11 and C17 versions except for the __STDC_VERSION__ macro, which expands to 201112L (for C11) and 201710L (for C17).

Here is an example that shows these features:

#include <assert.h> 
#include <stdalign.h> 
#include <stdlib.h> 
#include <stdnoreturn.h> 

#define NO_WARN(X)                                                  \
    _Pragma("warning (push)") _Pragma("warning (disable: 4146)") X; \
    _Pragma("warning (pop)") 

// Pick stored or storei based on the type of the dst 
#define store(x, y) _Generic((x),               \
                            data*: stored,      \
                            int* : storei)(x, y) 

typedef struct data {
    _Alignas(8) unsigned int i;
} data;

static_assert(alignof(data) == 8, "data is not properly aligned");

void stored(data* restrict dst, const data* restrict src)
{
    // Do not trigger warning 4245 
    dst->i = NO_WARN(-(src->i));

}

void storei(int* restrict dst, const int* restrict src)
{
    *dst = *src;
}


noreturn void my_exit(int ret) {
    exit(ret);
}

int main() {
    data src, dst;
    src.i = 5;

    int i, j;

    i = 10;

    store(&src, &dst);
    store(&i, &j);

    my_exit(0);
}

Since the inclusion of the token-based conformant preprocessor, the two new C compiler switches imply /Zc:preprocessor. If you want to use the traditional, character-based preprocessor alongside C11 or C17, you will need to pass in the /Zc:preprocessor- compiler switch explicitly. We do encourage you to fix your source code where needed to not depend on the legacy, traditional preprocessor.

What’s not:

While there is currently no support for any C11 optional features, we are committed to providing the most impactful optional features in future releases. Atomic and threading support are on our roadmap. Support for Complex numbers is currently not planned and their absence is enforced with the proper feature test macros. Please go to Visual Studio Developer Community and voice your support for features you wish to be implemented. We are looking for your input that we are making the correct prioritizations.
Due to the nature of the Windows heap, aligned_alloc support is missing. The alternative is to use _aligned_malloc.

Additionally, DR 400 support is currently unimplemented for realloc as this change would break ABI.

Variable Length Arrays

Astute readers will note that VLAs are also not supported. Variable length arrays are generally less efficient than comparable fixed sized arrays, and generally inefficient compared to equivalent malloc(), when safely and securely implemented. VLAs provide attack vectors comparable to those of the infamous gets() — deprecated and destined to removal — for opportunities of “shifting the stack” and other exploits. For these reasons we intend not to support VLAs as an optional feature in C11.

C11 and C17: Getting Started

In order to use C11 or C17 in your programs, the latest Windows SDK updates are required to work properly with the conforming preprocessor (/Zc:preprocessor), and the new Universal C Runtime. Windows SDK releases correspond with Windows OS releases. Since there has been no Windows release with these changes, you will need an Insider Preview Windows SDK – a preview version of the Windows SDK that corresponds with Windows builds currently being flighted for Windows Insiders. Note that after installing the Insider Preview Windows 10 SDK, Visual Studio projects configured to use the latest Windows SDK will use the Insider Preview.

Step 1: Log in with an Insider Microsoft Account

Anyone can create a free Microsoft account (https://signup.live.com/) and then opt into the Insider program. Simply go to https://insider.windows.com/en-us/for-developers and click ‘Register’ and log in.

After registering, you will be given the option to start flighting Insider versions of Windows, but this is not necessary to download and use the Insider Windows 10 SDK. Registering does not automatically opt in your machines to download new Windows flights.

Once you hit this page, you do not need to click ‘Flight now’. Continue to the next step and download the Insider Preview Windows 10 SDK.

Step 2: Download the Insider Preview Windows 10 SDK

The Insider Preview Windows SDK can be installed from https://www.microsoft.com/en-us/software-download/windowsinsiderpreviewSDK. If this is not the page you receive, make sure you are logged in with your Microsoft account that has been opted into being an Insider.

The Insider page describes that using a Windows 10 Insider Preview OS is required. This is true for some of the content included in the Windows SDK, as they may depend on new APIs not present on older versions of Windows, but the Windows and Universal C Runtime headers will properly install and be usable without an Insider OS.

Click ‘Get SDK Insider Preview – Build 20206’ to begin download. Future versions of the Windows SDK will also work.

Step 3: Install the Insider Preview Windows 10 SDK

The Insider Preview Windows SDK will download as an .iso file.

Mount the .iso file and run WinSDKSetup.exe to start installation.

Choose to install the Windows Software Development Kit on this computer and click next. You will have to choose whether to allow insights for Windows SDK usage and must accept the license agreement before you arrive at the feature installation page. The only features you should install (unless you are using an Insider Preview build of Windows 10) are:

  • Windows SDK Signing Tools for Desktop Apps
  • Windows SDK for UWP Managed Apps
  • Windows SDK for UWP C++ Apps
  • Windows SDK for Desktop C++ x86 Apps (if planning on building for x86)
  • Windows SDK for Desktop C++ amd64 Apps (if planning on building for amd64)
  • Windows SDK for Desktop C++ arm Apps (if planning on building for arm)
  • Windows SDK for Desktop C++ arm64 Apps (if planning on building for arm64)

The SDK will take a couple minutes to complete installation and then you can open Visual Studio 16.8 Preview 3.

Step 4: Configuring C11 or C17 mode in Visual Studio 16.8 Preview 3

Support for C11 and C17 begins in Visual Studio 16.8 Preview 3. You can download the latest Preview via https://visualstudio.microsoft.com/vs/preview/.

In your project, open the Properties page. First, we want to ensure that the project will use the Insiders Preview Window 10 SDK. Set Windows SDK Version to 10.0.20206.0 (or the latest Insider Preview Windows 10 SDK just installed).

You will also see a new option: C Language Standard. Set this to ISO C11 Standard (/std:c11) or ISO C17 (2018) Standard (/std:c17).

The C++ Language Standard is used when the language is C++ – this will be the default when the file extension is .cpp. The C Language Standard version is used when the language is C – this will be the default when the file extension is .c. To ensure the project is building with C11 or C17, you must ensure it is a .c file, or set the code to compile as C in the Properties tab.

After these project changes, you’re ready to test out C11 and C17!

In Conclusion

We are very excited to finally support C11 and C17! As the highest voted Developer Community suggestion, we’re sure you’re excited too. If you want support for the optional features of C11, please let us know on Developer Community.

As always, we welcome your feedback. We can be reached via the comments below or via email (visualcpp@microsoft.com). If you encounter problems with Visual Studio or MSVC, or have a suggestion for us, please let us know through Help > Send Feedback > Report A Problem / Provide a Suggestion in the product, or via Developer Community. You can also find us on Twitter (@VisualC).

Category
C++

Author

Elnar Dakeshov
Software Engineer 2

C++ front end developer :)

15 comments

Discussion is closed. Login to edit/delete existing comments.

  • Dusan Jovanovic

    Dear team decision-makers, please consider what have you missed on VLA's :

    1. VLA's have great use in simplifying declarations (stack not involved)
    https://gustedt.wordpress.com/2014/09/08/dont-use-fake-matrices/
    1.1 and generally VLA aided heap allocation, has great use for:
    https://gustedt.wordpress.com/2011/01/09/dont-be-afraid-of-variably-modified-types/

    2. VLA's and stack and security debate are done and dusted, a long time ago.
    Do we really have to have an artificial debate 6+ years after?
    Hint 1: the "other two" will never remove the vla.
    Hint 2: alloca() you dear team will never remove.

    It is great cl.exe will catchup with ISO C1x . It is now2020 but it is never...

    Read more
    • Vk TestA

      1. VLA’s are complicating compiler immensely. Especially if you have one compiler for C and C++ (for example “Universal references” in C++ suddenly stop being “universal” if you have VLAs).
      2. There are countless bugs related to VLAs both in clang and gcc related to VLAs.
      3. And you can make compiler C11/C17 compliant by omitting these and adding a single define: __STDC_NO_VLA__ (look it up in your copy of standard).

      Given #1, #2, and #3 it’s obvious why developers have picked what they have picked (C11/C17 with __STDC_NO_VLA__ define).

    • Vladimir Davidovich

      I’ve given up waiting for VLA support in MSVC. Switched to clang-cl, happy now.

      • Dusan Jovanovic

        There are some clang-cl.exe unwanted surprises;

        __attribute__((destructor))

        does not work with `/MD`, but it works with `/MT` ?

        Perhaps I was not digging enough, but it seems it is a moot point who do we talk to about `clang-cl.exe` issues?

      • Vk TestA

        clang-cl is part of clang and bugs should be submitted there.

        Microsoft just packages it and brings it with their IDE.

  • Royi Avital

    Thank you for that.
    It will be great if you added `/std:c99` as well.

    Please keep investing in the C compatibility.
    C is important to many of us.

    • Victor Khimenko

      I don’t think adding /std:c99 switch which doesn’t enable C99-compatible mode would be a good idea. And since VLAs are optional in C11 but required in C99… I think you should be happy with C11 and C17.

      • Dusan Jovanovic

        Is it not “cumulative”? C18 has all the features available in all the ISO standards before it.

        VLA is neither “optional” nor “required” it is in ISO C since C99. Please see the links I provided below.

        No VLA no ISO C, it is as simple as that 🙂

      • Vk TestA

        Look out for INCITS+ISO+IEC+9899-1999.pdf (if you have a license). Open page 80. Read the following example:

        #include
        size_t fsize3(int n)
        {
        char b[n+3]; // variable length array
        return sizeof b; // execution time sizeof
        }

        int main()
        {
        size_t size;
        size = fsize3(10); // fsize3 returns 13
        return 0;
        }

        If that's not a VLA then what it is? Even standard itself calls it variable length array!

        Or look https://en.wikipedia.org/wiki/C11_(C_standard_revision).

        Compiler without VLAs couldn't be called C99 compiler, sorry.

        C11 made these optional and added __STDC_NO_VLA__ macro for these. In C99 they were mandatory.

        Read more
      • F. v.S.

        VLA is made optional since C11 via WG14 N1460.

        You can download N1570 (C11 FDIS), N2176 (C17 FDIS), or any C2x working draft (e.g. N2583), and read 6.10.8.3 carefully.

        __STDC_NO_VLA__ The integer constant 1, intended to indicate that the implementation does not
        support variable length arrays or variably modified types.

        Read more
  • Richard Barrell

    This is delightful news!

    Will this include support for C99 designated initialiser syntax, please? I don’t see it explicitly mentioned above, though I’d assume it would be in since it’s older than C11 and doesn’t require anything from the runtime.

    Thank you.

    • Chuck Walbourn

      Alas, this was answered in an older post in the comments, but the comments were lost in the transition to devblogs.

      <code>

      Read more
    • Me Gusta

      As the post stated, they included all required features of C11. The features classed as optional are:
      Analyzability
      Bounds-checking interface
      Multithreading
      Atomic primitives and types
      IEC 60559 floating-point arithmetic
      IEC 60559 compatible complex arithmetic
      Complex types
      Varitable length arrays
      As you can see, designated initialisers are not in this list and are therefore classed as required.

    • Sunil Joshi

      I’m pretty sure that was already available in C mode but not C++ mode.

      • Curtis Bezault

        Designated initializers are available under /std:c++latest.