Take the 4-5 line Debian patch and turn it into six dozen lines of
new code. The submitted patch introduces use of several C library
routines that aren't presently in use, so would need testing by all
functional or nearly-functional ports to verify that it wouldn't
break anything. It also switched the formatted build date+time
from localtime to UTC. This makes the code conditional so it can
be ignored by anybody and avoid the risk of breakage. And a lot of
the increase in size is comments attempting to explain what the new
conditional is for: when REPRODUCIBLE_BUILD is defined, makedefs
will use getenv("SOURCE_DATE_EPOCH") (whose value is an integer
representing seconds since 1-Jan-1970) instead of current date+time
when generating date.h. The purpose is to be able to rebuild at a
later date and produce an identical program, which doesn't happen
when compile time gets incorporated into the binary.
I've added some sanity checking to try to make sure the getenv()
value obtained isn't bogus. And the version string put into date.h
will be slightly different, allowing someone who sees date.h or 'v'
output to tell whether SOURCE_DATE_EPOCH was involved: showing
"<port> NetHack <version> last revision <date>" instead of the
usual "... last build <date>".
To test, checkout a new branch for building, make any local edits
to unixconf.h and config.h, including enabling REPRODUCIBLE_BUILD,
git add+commit them, then use
SOURCE_DATE_EPOCH=`git log -1 --pretty=%ct` make install
Other ports will need a bit more work to set up the environment,
but can still use git to track file dates and supply the latest.
Building with alternate configurations could be accomplished by
using tags instead of 'log -1' or by using distinct build branches
where nothing is commited/merged/rebased after completed build.
Unresolved issue: BUILD_DATE, VERSION_ID, and COPYRIGHT_BANNER_C
contain formatted date+time but omit timezone. SOURCE_DATE_EPOCH
is assumed to be UTC but the formatted values don't say so, so it
might appear to be incorrect when compared with local time. We
definitely don't want to start mucking about with timezones within
nethack, so I think we just live with this. It's not an issue for
default configruation where REPRODUCIBLE_BUILD is left disabled.
357 lines
8.9 KiB
C
357 lines
8.9 KiB
C
/* NetHack 3.6 version.c $NHDT-Date: 1449328116 2015/12/05 15:08:36 $ $NHDT-Branch: NetHack-3.6.0 $:$NHDT-Revision: 1.41 $ */
|
|
/* Copyright (c) Stichting Mathematisch Centrum, Amsterdam, 1985. */
|
|
/* NetHack may be freely redistributed. See license for details. */
|
|
|
|
#include "hack.h"
|
|
#include "dlb.h"
|
|
#include "date.h"
|
|
/*
|
|
* All the references to the contents of patchlevel.h have been moved
|
|
* into makedefs....
|
|
*/
|
|
#ifdef SHORT_FILENAMES
|
|
#include "patchlev.h"
|
|
#else
|
|
#include "patchlevel.h"
|
|
#endif
|
|
|
|
#define BETA_INFO ""
|
|
|
|
STATIC_DCL void FDECL(insert_rtoptions, (winid,char *,const char *));
|
|
|
|
/* fill buffer with short version (so caller can avoid including date.h) */
|
|
char *
|
|
version_string(buf)
|
|
char *buf;
|
|
{
|
|
return strcpy(buf, VERSION_STRING);
|
|
}
|
|
|
|
/* fill and return the given buffer with the long nethack version string */
|
|
char *
|
|
getversionstring(buf)
|
|
char *buf;
|
|
{
|
|
Strcpy(buf, VERSION_ID);
|
|
#if defined(BETA) && defined(BETA_INFO)
|
|
Sprintf(eos(buf), " %s", BETA_INFO);
|
|
#endif
|
|
#if defined(RUNTIME_PORT_ID)
|
|
append_port_id(buf);
|
|
#endif
|
|
return buf;
|
|
}
|
|
|
|
/* the 'v' command */
|
|
int
|
|
doversion()
|
|
{
|
|
char buf[BUFSZ];
|
|
|
|
pline("%s", getversionstring(buf));
|
|
return 0;
|
|
}
|
|
|
|
/* the '#version' command; also a choice for '?' */
|
|
int
|
|
doextversion()
|
|
{
|
|
dlb *f;
|
|
char *pd, buf[BUFSZ];
|
|
winid win = create_nhwindow(NHW_TEXT);
|
|
boolean rtadded = FALSE;
|
|
|
|
/* instead of using ``display_file(OPTIONS_USED,TRUE)'' we handle
|
|
the file manually so we can include dynamic version info */
|
|
putstr(win, 0, getversionstring(buf));
|
|
|
|
f = dlb_fopen(OPTIONS_USED, "r");
|
|
if (!f) {
|
|
putstr(win, 0, "");
|
|
Sprintf(buf, "[Configuration '%s' not available?]", OPTIONS_USED);
|
|
putstr(win, 0, buf);
|
|
} else {
|
|
/*
|
|
* already inserted above:
|
|
* + outdented program name and version plus build date and time
|
|
* dat/options; display the contents with lines prefixed by '-'
|
|
* deleted:
|
|
* - blank-line
|
|
* - indented program name and version
|
|
* blank-line
|
|
* outdented feature header
|
|
* - blank-line
|
|
* indented feature list
|
|
* spread over multiple lines
|
|
* blank-line
|
|
* outdented windowing header
|
|
* - blank-line
|
|
* indented windowing choices with
|
|
* optional second line for default
|
|
* - blank-line
|
|
* - EOF
|
|
*/
|
|
boolean prolog = TRUE; /* to skip indented program name */
|
|
|
|
while (dlb_fgets(buf, BUFSZ, f)) {
|
|
(void) strip_newline(buf);
|
|
if (index(buf, '\t') != 0)
|
|
(void) tabexpand(buf);
|
|
|
|
if (*buf && *buf != ' ') {
|
|
/* found outdented header; insert a separator since we'll
|
|
have skipped corresponding blank line inside the file */
|
|
putstr(win, 0, "");
|
|
prolog = FALSE;
|
|
}
|
|
/* skip blank lines and prolog (progame name plus version) */
|
|
if (prolog || !*buf)
|
|
continue;
|
|
|
|
if (!rtadded) {
|
|
const char *catchphrase = ", and basic NetHack features.";
|
|
|
|
pd = strstri(buf, catchphrase);
|
|
if (pd) {
|
|
*pd = '\0';
|
|
insert_rtoptions(win, buf, &catchphrase[2]);
|
|
rtadded = TRUE; /* only do it once */
|
|
}
|
|
}
|
|
if (*buf)
|
|
putstr(win, 0, buf);
|
|
}
|
|
(void) dlb_fclose(f);
|
|
display_nhwindow(win, FALSE);
|
|
destroy_nhwindow(win);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
extern const char regex_id[];
|
|
|
|
static const char *rt_opts[] = {
|
|
"pattern matching via", regex_id,
|
|
};
|
|
static const char indent[] = " ";
|
|
|
|
/*
|
|
* 3.6.0
|
|
* Some optional stuff is no longer available to makedefs because
|
|
* it depends which of several object files got linked into the
|
|
* game image, so we insert those options here.
|
|
*/
|
|
STATIC_OVL void
|
|
insert_rtoptions(win, buf, finalphrase)
|
|
winid win;
|
|
char *buf;
|
|
const char *finalphrase;
|
|
{
|
|
char rtbuf[BUFSZ];
|
|
int l, i;
|
|
const char *s1 = 0, *s2 = 0, *s3 = 0, *s4 = 0;
|
|
|
|
if ((int) strlen(buf) >= (BUFSZ - 1))
|
|
return;
|
|
|
|
strcpy(rtbuf, buf);
|
|
for (i = 0; i < (SIZE(rt_opts) + 1); i += 2) {
|
|
if (i < SIZE(rt_opts)) {
|
|
s1 = ", ";
|
|
s2 = rt_opts[i];
|
|
s3 = " ";
|
|
s4 = rt_opts[i+1];
|
|
} else {
|
|
s1 = " ";
|
|
s2 = finalphrase;
|
|
s3 = "";
|
|
s4 = "";
|
|
}
|
|
l = (int) strlen(rtbuf) + (int) strlen(s1) + (int) strlen(s2)
|
|
+ (int) strlen(s3) + (int) strlen(s4) + 1;
|
|
if (l <= (COLNO - 5) && l < (BUFSZ - 1)) {
|
|
Strcat(rtbuf, s1);
|
|
Strcat(rtbuf, s2);
|
|
Strcat(rtbuf, s3);
|
|
Strcat(rtbuf, s4);
|
|
} else {
|
|
putstr(win, 0, rtbuf);
|
|
if (i >= SIZE(rt_opts))
|
|
s1 = "";
|
|
l = (int) strlen(indent) + (int) strlen(s1) + (int) strlen(s2)
|
|
+ (int) strlen(s3) + (int) strlen(s4) + 1;
|
|
if (l <= (COLNO - 5) && l < (BUFSZ - 1)) {
|
|
Strcpy(rtbuf, indent);
|
|
Strcat(rtbuf, s1);
|
|
Strcat(rtbuf, s2);
|
|
Strcat(rtbuf, s3);
|
|
Strcat(rtbuf, s4);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (l)
|
|
putstr(win, 0, rtbuf);
|
|
*buf = '\0';
|
|
}
|
|
|
|
#ifdef MICRO
|
|
boolean
|
|
comp_times(filetime)
|
|
long filetime;
|
|
{
|
|
/* BUILD_TIME is constant but might have L suffix rather than UL;
|
|
'filetime' is historically signed but ought to have been unsigned */
|
|
return (boolean) ((unsigned long) filetime < (unsigned long) BUILD_TIME);
|
|
}
|
|
#endif
|
|
|
|
boolean
|
|
check_version(version_data, filename, complain)
|
|
struct version_info *version_data;
|
|
const char *filename;
|
|
boolean complain;
|
|
{
|
|
if (
|
|
#ifdef VERSION_COMPATIBILITY
|
|
version_data->incarnation < VERSION_COMPATIBILITY
|
|
|| version_data->incarnation > VERSION_NUMBER
|
|
#else
|
|
version_data->incarnation != VERSION_NUMBER
|
|
#endif
|
|
) {
|
|
if (complain)
|
|
pline("Version mismatch for file \"%s\".", filename);
|
|
return FALSE;
|
|
} else if (
|
|
#ifndef IGNORED_FEATURES
|
|
version_data->feature_set != VERSION_FEATURES
|
|
#else
|
|
(version_data->feature_set & ~IGNORED_FEATURES)
|
|
!= (VERSION_FEATURES & ~IGNORED_FEATURES)
|
|
#endif
|
|
|| version_data->entity_count != VERSION_SANITY1
|
|
|| version_data->struct_sizes1 != VERSION_SANITY2
|
|
|| version_data->struct_sizes2 != VERSION_SANITY3) {
|
|
if (complain)
|
|
pline("Configuration incompatibility for file \"%s\".", filename);
|
|
return FALSE;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
/* this used to be based on file date and somewhat OS-dependant,
|
|
but now examines the initial part of the file's contents */
|
|
boolean
|
|
uptodate(fd, name)
|
|
int fd;
|
|
const char *name;
|
|
{
|
|
int rlen;
|
|
struct version_info vers_info;
|
|
boolean verbose = name ? TRUE : FALSE;
|
|
|
|
rlen = read(fd, (genericptr_t) &vers_info, sizeof vers_info);
|
|
minit(); /* ZEROCOMP */
|
|
if (rlen == 0) {
|
|
if (verbose) {
|
|
pline("File \"%s\" is empty?", name);
|
|
wait_synch();
|
|
}
|
|
return FALSE;
|
|
}
|
|
if (!check_version(&vers_info, name, verbose)) {
|
|
if (verbose)
|
|
wait_synch();
|
|
return FALSE;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
void
|
|
store_version(fd)
|
|
int fd;
|
|
{
|
|
static const struct version_info version_data = {
|
|
VERSION_NUMBER, VERSION_FEATURES,
|
|
VERSION_SANITY1, VERSION_SANITY2, VERSION_SANITY3
|
|
};
|
|
|
|
bufoff(fd);
|
|
/* bwrite() before bufon() uses plain write() */
|
|
bwrite(fd, (genericptr_t) &version_data,
|
|
(unsigned) (sizeof version_data));
|
|
bufon(fd);
|
|
return;
|
|
}
|
|
|
|
#ifdef AMIGA
|
|
const char amiga_version_string[] = AMIGA_VERSION_STRING;
|
|
#endif
|
|
|
|
unsigned long
|
|
get_feature_notice_ver(str)
|
|
char *str;
|
|
{
|
|
char buf[BUFSZ];
|
|
int ver_maj, ver_min, patch;
|
|
char *istr[3];
|
|
int j = 0;
|
|
|
|
if (!str)
|
|
return 0L;
|
|
str = strcpy(buf, str);
|
|
istr[j] = str;
|
|
while (*str) {
|
|
if (*str == '.') {
|
|
*str++ = '\0';
|
|
j++;
|
|
istr[j] = str;
|
|
if (j == 2)
|
|
break;
|
|
} else if (index("0123456789", *str) != 0) {
|
|
str++;
|
|
} else
|
|
return 0L;
|
|
}
|
|
if (j != 2)
|
|
return 0L;
|
|
ver_maj = atoi(istr[0]);
|
|
ver_min = atoi(istr[1]);
|
|
patch = atoi(istr[2]);
|
|
return FEATURE_NOTICE_VER(ver_maj, ver_min, patch);
|
|
/* macro from hack.h */
|
|
}
|
|
|
|
unsigned long
|
|
get_current_feature_ver()
|
|
{
|
|
return FEATURE_NOTICE_VER(VERSION_MAJOR, VERSION_MINOR, PATCHLEVEL);
|
|
}
|
|
|
|
/*ARGUSED*/
|
|
const char *
|
|
copyright_banner_line(indx)
|
|
int indx;
|
|
{
|
|
#ifdef COPYRIGHT_BANNER_A
|
|
if (indx == 1)
|
|
return COPYRIGHT_BANNER_A;
|
|
#endif
|
|
#ifdef COPYRIGHT_BANNER_B
|
|
if (indx == 2)
|
|
return COPYRIGHT_BANNER_B;
|
|
#endif
|
|
#ifdef COPYRIGHT_BANNER_C
|
|
if (indx == 3)
|
|
return COPYRIGHT_BANNER_C;
|
|
#endif
|
|
#ifdef COPYRIGHT_BANNER_D
|
|
if (indx == 4)
|
|
return COPYRIGHT_BANNER_D;
|
|
#endif
|
|
return "";
|
|
}
|
|
|
|
/*version.c*/
|