The status is a 16-bit word (something like JSON, but for the 1970s where a 16-bit register was a rare commodity, especially after something complicated like fork) wherein various bits or ranges of bits have particular meanings. Perl puts the 16-bit word set by a wait(2) call into the $? variable, and we can find the particular exit code by shifting the upper bits over, because that is where that field is in the word:
$ perl -e 'fork || exit 42; wait; printf "%016b\n", $?'
0010101000000000
$ perl -e 'fork || exit 42; wait; printf "%016b\n", $? >> 8'
0000000000101010
$ perl -e 'fork || exit 42; wait; printf "%d\n", $? >> 8'
42
Unix shells mangle the 16-bit status word into a different form:
$ perl -e 'exit 42'; echo $?
42
which becomes relevant if there was a core dump (there's a flag for that in the 16-bit word) or a signal:
$ perl -MPOSIX -e 'raise SIGTERM'; echo $?
Terminated
143
$ echo $((143-128))
15
$ kill -l | grep 15
15   TERM Terminated                    31   USR2 User defined signal 2
So that's the signal number plus 128 (the shell mangling the value) which in the actual 16-bit exit status word instead appears as:
$ perl -MPOSIX -e 'fork || raise SIGTERM; wait; printf "%016b\n", $?'
0000000000001111
$ perl -E 'say 0b1111'
15
So Perl is using the 16-bit word returned by a wait(2) call directly, while the Unix shell mangles it into some other form. Depending on the language you are using, you'll need to either follow the C convention and do things like "left shift to get the exit code" (there are often macros or other such calls to assist with this, check the fine documentation for details), or "subtract 128 to find the signal number" if using the shell mangled value.
For extra points, create a program that creates a core file and see what the exit status word looks like, or try different signal numbers, etc.
How To Use The Status Word
There are various conventions which may or may not be frowned upon, e.g. the exit codes from sysexits.h; some of these codes are important to Email delivery systems (by way of sendmail, historically) where particular codes can be used to signal a "do try again" temporary failure (which becomes a 4xx code in SMTP), as opposed to a "reject and bounce that email" (5xx) which you may see references to in mail delivery agent software such as procmail or similar.
$ grep EX_TEMPFAIL /usr/include/sysexits.h
 *      EX_TEMPFAIL -- temporary failure, indicating something that
#define EX_TEMPFAIL     75      /* temp failure; user is invited to retry */
Otherwise, 0 usually means "okay" (but programs may fail without setting a non-zero code, buyer beware!) and anything not-0 means "not okay" (but what exactly that means varies, it could be a temporary failure, noise to ignore, something really really bad, etc). There could be site or vendor or software specific meanings for specific exit codes; these particular codes may or may not be documented. A man page may commonly have a generic .Ex -std macro to generate a blurb like
EXIT STATUS
     The foo utility exits 0 on success, and >0 if an error occurs.
or more specific documentation as need be, if the software is well documented. A lot of software isn't so well documented...
When Is This Important
Monitoring or test suits probably do need to distinguish between "any old non-zero exit" and a program suddenly instead exiting with a segfault or coredump, or perhaps a segfault is the expected value for a particular test, and it would be unusual for that not to happen. Monitoring or tests that only report 0-or-not will omit valuable debugging information.
SMTP servers, as mentioned above, will perform different actions depending on the precise exit status word returned by a local delivery agent. A hard bounce is very different from a temporary delivery failure that the server will likely try again.
     
    
errnoreturned by various functions within the GNU C libraries. There is no logical connection between these error codes and any process exit convention. Furthermore, this list overlaps the range which the article itself specifies as used by the shell (120 to 127), and also overlaps the values returned when a process is terminated by signals (e.g. 130 for SIGINT). In practice, I believe no standard utility returns an exit code above 5.