我在圣诞节期间作的演讲「DWARF 眼中的 C++」,介绍了 DWARF 关联 C++ 程序的语义元素、文本、和代码生成的基础知识。视频链接在 Speaker Deck 页面上。
https://speakerdeck.com/lichray/dwarf-yan-zhong-de-c-plus-plus
Defer available in gcc and clang
About a year ago I posted about defer and that it would be available for everyone using gcc and/or clang soon. So it is probably time for an update.
Two things have happened in the mean time:
I have not yet got my hands on the gcc implementation (but this is also less urgent, see below), but I have been able to use clang’s which is available starting with clang-22.
I think that with this in mind everybody developing in C could and should now seriously consider switching to defer for their cleanup handling:
I am not sure if the compiler people are also planning back ports of these features, but with some simple work around and slightly reduced grammar for the defer feature this works for me from gcc-9 onward and for clang-22 onward:
So this is already a large panel of compilers. Obviously it depends on your admissible compile platforms whether or not these are sufficient for you. In any case, with these you may compile for a very wide set of installs since defer does not need any specific software infrastructure or library once the code is compiled.
As already discussed many times, the gcc fallback uses the so-called “nested function” feature which is always subject of intense debate and even flame wars. Don’t worry, the implementation as presented here, even when compiled with no optimization at all, does not produce any hidden function in the executable, and in particular there is no “trampoline” or whatever that would put your execution at risk of a stack exploit.
You may also notice that there is no fallback for older clang version. This is because their so-called “blocks” extension cannot easily be used as a drop-in to replace nested function: their semantics to access variables from the surrounding scope are different and not compatible with the defer feature as defined by TS 25755.
So for example if you are scared of using big VLA on the stack, you may use the above code in headers and something like
double* BigArray = malloc(sizeof(double[aLot])); if (!BigArray { exit(EXIT_FALURE); } defer { free(BigArray); }to have an implementation of a big array with a failure mode for the allocation.
Or if you want to be sure that all your mutexes are unlocked when you leave a critical section, use and idiom as here
{ if (mtx_lock(&mtx) != thrd_success) { exit(EXIT_FAILURE); } defer { mtx_unlock(&mtx); } ... do something complicated ... if (rareCondition) { return 42; } ... do something even more complicated ... }Just notice, that you’d always have to use the defer feature with curly braces to ensure that the gcc fallback works smoothly.
OMG I just found that WinDbg now supports using DWARF from PE files and that also work with minidumps. That means I can ship binary built from x86_64-pc-windows-gnu and still be able to troubleshoot crashes in the field. It means I don't have to deal with vcredist. It means my contributor don't have to install VC buildtools. 😀
1. https://learn.microsoft.com/en-us/windows-hardware/drivers/debugger/linux-dwarf-symbols
2. https://github.com/microsoft/WinDbg-Feedback/issues/338
RE: https://hachyderm.io/@fasterthanlime/115730529166857536
It's useless to merely condemn GitHub morally. As long as startups like Blacksmith continues to grow up, such behavior by GitHub/Microsoft will remain difficult to stop :/