Bug #10363
closedInvalid implicit conversion from void to int
0%
Description
Although Frama-C fails for the attached source file, GCC can compile it without critical errors. There is such the error:
[kernel] /home/debian/klever-inst/klever/build bases/build-base-linux-4.18-x86_64-allmodconfig/Storage/var/lib/klever/workspace/Branches-and-Tags-Processing/linux-4.18/drivers/block/paride/frpw.c:55:8: Failure: invalid implicit conversion from void to int [kernel] User Error: skipping file /home/debian/klever-inst/klever-work/native-scheduler/scheduler/jobs/7292d346-219c-4d78-b695-7712a530fff6/klever-core-work-dir/job/vtg/drivers/block/paride/frpw.ko/empty/weaver/frpw.c that has errors. [kernel] Frama-C aborted: invalid user input.
Files
Updated by Evgeny Novikov almost 3 years ago
I revealed this issue one more time, but now I built up a simple example:
int main(int argc, char *argv[])
{
int res;
res = (argc > 1 ? ({ (void)1; }) : (void)2, (int)3);
return res;
}
You can compile this code and check that function main always returns 3, while CIL fails to parse it with the error as in the issue description. It looks like the internal compound statement "({ ... })" confuses CIL. If you will replace it with, say, "(void)1" it will work.
This is example is artificial. The original issue comes from substitution of macro r1(), say, in https://elixir.bootlin.com/linux/v5.5.19/source/drivers/block/paride/friq.c#L58. Hopefully there are just several drivers that do this.
Updated by Mikhail Mandrykin almost 3 years ago
- Status changed from New to Open
Attempted a tentative fix in framac|34aba90f5efb. The reason for the failure is, in particular, the line TVoid _ when what' = ADrop -> (t', e') (* strange GNU thing *)
in handling of casts that seems to be intentionally added to work-around some GNU incompatibility. In certain cases when the value of an expression is statically ignored (as in this case due to the ,
operator), the outer cast to void
is dropped for some reason. Why this was at all necessary at some point is not clear to me, especially since the line was in the codebase starting right from the initial commit. So the fix just attempts to minimize the impact of this workaround by taking the constant/non-constant context into consideration: We disable the [strange GNU] compatibility in non-constant contexts. Additionally, it was necessary to allow conditional conversions between two void
types
Updated by Evgeny Novikov almost 3 years ago
- Status changed from Open to Resolved
Thank you, I will test this and report results in a while.
Updated by Evgeny Novikov almost 3 years ago
- File i7300_idle.i.tar.gz i7300_idle.i.tar.gz added
- Status changed from Resolved to Open
It seems that the fix is not valid in some cases. For instance, after running new Frama-C for the attached input file:
toplevel.opt "-no-autoload-plugins" "-no-findlib" "-kernel-log" "e:problem desc.txt" "-machdep" "gcc_x86_64" "-c11" "-kernel-warn-key" "CERT:MSC:38=active" "-remove-unused-inline-functions" "-remove-unused-static-functions" "-no-annot" "-no-single-return" "-fold-temp-vars" "-shrink-initializers" "-keep-logical-operators" "-aggressive-merging" "-print" "-print-lines" "-no-print-annot" "-ocode" "cil.i" "i7300_idle.i"
I got the invalid CIL file containing several void temporary variables. GCC and CPAchecker fails to parse it:
$ gcc -fsyntax-only -x c cil.i .../i7300_idle.c: In function ‘i7300_idle_notifier’: .../i7300_idle.c:463:12: error: variable or field ‘tmp’ declared void ...
Updated by Mikhail Mandrykin almost 3 years ago
Ultimately reverted the framac|34aba90f and pushed an entirely different fix framac|8caa89de. Basically, instead of disabling the workaround in this case, now the solution is to actually reproduce the work-around even in case of a GNU-body ({ ...; (void) e; }
), where the what = ADrop
cannot be passed from the caller and so the special handling is added in yet another place (GNU-body handling). The possible side-effect is translating some extra code in the ignored GNU bodies that previously were entirely dropped, which potentially can expose some more bugs
Updated by Evgeny Novikov almost 3 years ago
- File add_disk.i.tar.gz add_disk.i.tar.gz added
The quite large test suit detected only one issue. Frama-C fails at processing the attached source file with the following error:
[kernel] .../linux-stable/include/linux/rhashtable.h:616:4: Failure: lvalue of type void: tmp_4
This source file could be handled by GCC without errors.
Updated by Evgeny Novikov almost 3 years ago
- File epia.i.tar.gz epia.i.tar.gz added
Besides, I checked that even though the issue has gone for the simplified test case, it still exists for the original complicated source file. I attached the newer version. The corresponding error is as follows:
[kernel] .../linux-5.10.27/drivers/block/paride/epia.c:59:10: Failure: invalid implicit conversion from void to int
Updated by Mikhail Mandrykin almost 3 years ago
Pushed framac|5b8e240a. It just refines treatment of some void
expressions within GNU_BODY and in the arguments of the ?:
operator. Inside the body we seek to avoid creating temp vars if the result is dropped; inside the ?:
operator we allow `void` expressions in either side if the result is dropped. The attached test cases now pass, however, in general the semantics of the fixes is quite lax. In particular, epia.c
has a call to __udelay
and (void) 0
as arguments to ?:
, which makes calculating the resulting type of the operation dubious: it can be either naively treated as `void` (which is wrong if the expression is later used in some larger context, as a temporary of type void
is prohibited) or (void) 0
can be heuristically treated as plain 0
of type int
(as in the heuristic discussed earlier, which is wrong since an implicit conversion between `int` and `void` is also prohibited). Technically, it seems there should be no expressions of type void
in C, but the GNU {...; statement }
allows for such possibility, hence all the issues
Updated by Mikhail Mandrykin almost 3 years ago
Partially reverted previous fixes in favor of a minimal change: framac|93c99dbf. Allowing an implicit cast between `void` and any type in case of an ignored result covers all the cases. I finally (after attempting many manual tests with GCC, LLVM and Frama-C) vaguely understand the logic behind all those cases: if the final result is ultimately dropped, a sub-expression of type void
is treated as if it is of type int
with all the corresponding implicit conversions or even as an expression of any type. However, the notion of "ultimately" is not very consistent: Frama-C accepted a computation (int)(x ? f() : (char *)0);
(f
is void
) even before the patch, while GCC and LLVM failed on that, but a simpler x ? f() : (char *)0
is accepted by all of them. Also, x ? f() : (struct s) {.f = 0 }
is accepted by LLVM and GCC, but not Frama-C (even after the minimal fix since the ignored call is treated as (int
)0
elsewhere). The problem here is properly propagating the static approximation of the dropped
flag, which is already different in Frama-C vs. GCC and LLVM. A really thorough fix would require consistently changing it everywhere, which is a lot of places in the entire expression translation function (about 2000 lines of code)
Updated by Mikhail Mandrykin almost 3 years ago
After more tests: Actually, the behavior of GCC and LLVM can be explained simpler: the result of a conditional conversion between void
and any other type is simply void
. Then the value of type void
must be ignored (other conversions and operations don't allow it). No dropped
flags needed anywhere. However, in Frama-C there are many places implicitly treating ignored expressions as int
(the dummy value 0
): at least in function calls, in casts, in statement expressions. In general this still requires a thorough revamp of the overall approach in the entire Frama-C translation procedure (the ubiquitous what
argument is probably just a generally wrong optimization)
Updated by Evgeny Novikov almost 3 years ago
- Status changed from Open to Closed
Despite Mikhail pointed out that there is still much work to do due to mismatches between compilers and Frama-C, his fixes helped to overcome the specified issue and there is no degradations. The considerable refactoring of Frama-C in this regard is a separate topic.