From 9a7db96696ce52292cf5d3d6aa6c8dcd1a9a8744 Mon Sep 17 00:00:00 2001 From: Roy Ivy III Date: Wed, 24 Apr 2019 10:38:40 -0500 Subject: [PATCH] Improve process exit operation .# Discussion `exit EXITCODE` and `exit /B EXITCODE` do NOT return EXITCODE as the process exit code unless the batch file is executed with a `call BAT_FILE` command. The ERRORLEVEL *is* set and returned to the parent shell correctly, but the process exit code *is not*. For example, given a BAT file "naive-exit-with-error.BAT": ``` batch :: naive-exit-with-error.BAT @exit /B -1 ``` The shell command `call naive-exit-with-error.bat || echo PROCESS_ERROR` will correctly print "PROCESS_ERROR". However, `naive-exit-with-error.bat || echo PROCESS_ERROR` will *never* print "PROCESS_ERROR". This is obviously not controllable by the batch file itself. So, another method must be used to force both the correct ERRORLEVEL and process exit code. Notably, a known property of `goto` when jumping to a nonexistent label can be used to overcome the issue. Per Dave Benham (aka, @dbenham), "GOTO undefined label is a fatal error. It immediately terminates batch processing, but the currently executing block of code that has already been parsed continues to completion. However, the remainder of the code is executed using a command line context instead of a batch context." This property can be used as in this example: ``` batch set "exit_val=%ERRORLEVEL%" goto __nonexistent_label_to_shift_into_command_line_context__ 2>NUL || "%COMSPEC%" /c exit %exit_val%` ``` This batch file code will always return the ERRORLEVEL and a correct corresponding process exit code, no matter how the batch file is called. .#### refs [1] @dbenham's comment on the answer [`@`](http://archive.is/6CgpF) > GOTO undefined label is a fatal error. It immediately terminates batch processing, but the currently executing block of code that has already been parsed continues to completion. However, the remainder of the code is executed using a command line context instead of a batch context. [2] [`@`](https://archive.is/X2mkm) [3] [`@`](https://archive.is/X2mkm) [4] [`@`](https://archive.is/aXJRz) --- lib/ExtUtils/PL2Bat.pm | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lib/ExtUtils/PL2Bat.pm b/lib/ExtUtils/PL2Bat.pm index ab06ca3..caeb65c 100644 --- a/lib/ExtUtils/PL2Bat.pm +++ b/lib/ExtUtils/PL2Bat.pm @@ -47,7 +47,7 @@ EOT my $tail = <<'EOT'; __END__ :endofperl - @set "ErrorLevel=" & @"%COMSPEC%" /d/c @exit %ErrorLevel% + @set "ErrorLevel=" & @goto _undefined_label_ 2>NUL || @"%COMSPEC%" /d/c @exit %ErrorLevel% EOT $tail =~ s/^\s+//gm;