Files
neovim/src/nvim/version.c
Florian Walch 8130eb1191 jemalloc: Force use of prefixed functions.
* Set JEMALLOC_NO_DEMANGLE to be able to use `je_*` functions,
  regardless of how jemalloc was compiled (--with-jemalloc-prefix)
* Show jemalloc information in Neovim's version output.

Resolve #2449.
2015-04-29 22:26:55 +03:00

1130 lines
16 KiB
C

/// @file version.c
///
/// Nvim was forked from Vim 7.4.160.
/// Vim originated from Stevie version 3.6 (Fish disk 217) by GRWalter (Fred).
#include <inttypes.h>
#include <assert.h>
#include <limits.h>
#include "nvim/vim.h"
#include "nvim/ascii.h"
#include "nvim/iconv.h"
#include "nvim/version.h"
#include "nvim/charset.h"
#include "nvim/memline.h"
#include "nvim/memory.h"
#include "nvim/message.h"
#include "nvim/misc2.h"
#include "nvim/screen.h"
#include "nvim/strings.h"
// version info generated by the build system
#include "auto/versiondef.h"
#define STR_(x) #x
#define STR(x) STR_(x)
// for the startup-screen ( ":intro" command )
#define NVIM_VERSION_MEDIUM STR(NVIM_VERSION_MAJOR) "." STR(NVIM_VERSION_MINOR)
// for the ":version" command and "nvim --version"
#define NVIM_VERSION_LONG "NVIM " NVIM_VERSION_MEDIUM "." STR(NVIM_VERSION_PATCH) NVIM_VERSION_PRERELEASE NVIM_VERSION_BUILD
char *Version = VIM_VERSION_SHORT;
char *longVersion = NVIM_VERSION_LONG;
char *longVersionWithDate = NVIM_VERSION_LONG " (compiled " __DATE__ " " __TIME__ ")";
char *mediumVersion = NVIM_VERSION_MEDIUM;
char *version_commit = "Commit: " NVIM_VERSION_COMMIT;
char *version_buildtype = "Build type: " NVIM_VERSION_BUILD_TYPE;
char *version_cflags = "Compilation: " NVIM_VERSION_CFLAGS;
#ifdef INCLUDE_GENERATED_DECLARATIONS
# include "version.c.generated.h"
#endif
static char *features[] = {
#ifdef HAVE_ACL
"+acl",
#else
"-acl",
#endif
#if (defined(HAVE_ICONV_H) && defined(USE_ICONV)) || defined(DYNAMIC_ICONV)
# ifdef DYNAMIC_ICONV
"+iconv/dyn",
# else
"+iconv",
# endif
#else
"-iconv",
#endif
#ifdef HAVE_JEMALLOC
"+jemalloc",
#else
"-jemalloc",
#endif
NULL
};
// clang-format off
static int included_patches[] = {
712,
711,
710,
//709,
//708,
//707,
//706,
//705,
//704,
//703,
702,
//701,
//700,
699,
//698,
//697,
//696,
//695,
//694,
//693,
//692,
//691,
690,
//689,
//688,
//687,
//686,
//685,
//684,
//683,
//682,
//681,
//680,
//679,
//678,
//677,
//676,
//675,
//674,
673,
//672,
//671,
//670,
//669,
//668,
//667,
//666,
//665,
//664,
//663,
//662,
//661,
660,
//659,
//658,
//657,
//656,
//655,
//654,
//653,
//652,
//651,
//650,
//649,
//648,
//647,
//646,
//645,
//644,
//643,
//642,
//641,
//640,
//639,
//638,
637,
636,
//635,
//634,
633,
//632,
//631,
630,
//629,
//628,
//627,
//626,
//625,
//624,
//623,
//622,
//621,
//620,
//619,
//618,
//617,
//616,
615,
//614,
//613,
//612,
//611,
//610,
//609,
//608,
//607,
//606,
//605,
//604,
//603,
//602,
601,
//600,
//599,
//598,
//597,
//596,
595,
//594,
//593,
//592,
//591 NA
//590,
//589 NA
588,
//587,
//586 NA
585,
//584 NA
//583 NA
//582,
//581,
580,
//579,
578,
//577,
//576,
//575,
//574,
//573,
572,
//571 NA
//570 NA
//569,
568,
567,
566,
565,
//564,
563,
//562,
561,
//560 NA
559,
//558 NA
//557 NA
//556 NA
//555 NA
554,
553,
552,
551,
550,
549,
//548 NA
547,
546,
545,
//544 NA
543,
542,
541,
//540 NA
539,
538,
537,
536,
//535,
//534 NA
533,
532,
//531,
//530,
529,
528,
527,
//526,
525,
//524,
//523 NA
//522 NA
521,
520,
519,
518,
517,
516,
515,
514,
513,
//512 NA
//511 NA
//510 NA
//509 NA
508,
//507 NA
//506 NA
//505 NA
//504 NA
503,
502,
//501 NA
500,
499,
//498 NA
497,
//496 NA
//495 NA
494,
493,
492,
491,
490,
489,
488,
487,
486,
485,
//484 NA
483,
//482 NA
//481 NA
//480 NA
//479 NA
478,
477,
//476 NA
//475 NA
474,
473,
472,
//471 NA
470,
//469 NA
468,
467,
466,
//465 NA
//464 NA
463,
462,
//461 NA
//460 NA
//459 NA
458,
457,
456,
455,
454,
//453 NA
452,
//451 NA
450,
449,
//448 NA
447,
446,
//445,
444,
//443 NA
442,
441,
440,
439,
438,
437,
436,
435,
434,
433,
//432 NA
//431 NA
//430 NA
//429 NA
//428 NA
427,
//426 NA
425,
//424 NA
423,
422,
421,
//420 NA
419,
418,
417,
416,
415,
414,
//413 NA
//412 NA
411,
410,
//409 NA
408,
407,
406,
405,
//404 NA
//403 NA
//402 NA
//401 NA
//400 NA
//399 NA
//398 NA
397,
//396,
395,
//394 NA
//393 NA
392,
391,
390,
//389,
388,
387,
386,
//385 NA
//384 NA
383,
382,
381,
//380 NA
379,
378,
377,
376,
375,
374,
373,
372,
371,
370,
369,
368,
367,
//366 NA
365,
364,
//363 NA
362,
361,
360,
359,
358,
357,
//356 NA
355,
//354 NA
353,
352,
351,
350,
349,
348,
347,
346,
345,
344,
343,
//342 NA
341,
//340 NA
339,
338,
337,
336,
335,
334,
//333 NA
//332 NA
331,
330,
329,
328,
327,
//326 NA
325,
324,
323,
//322 NA
//321 NA
320,
//319 NA
318,
317,
//316 NA
315,
314,
313,
312,
311,
310,
309,
308,
//307 NA
306,
305,
//304 NA
303,
302,
301,
//300 NA
//299 NA
298,
297,
296,
295,
294,
293,
292,
291,
290,
289,
288,
//287 NA
286,
285,
284,
//283 NA
282,
281,
280,
279,
278,
277,
276,
275,
274,
//273 NA
272,
//271 NA
//270 NA
269,
268,
267,
266,
265,
264,
//263 NA
262,
261,
260,
//259 NA
//258 NA
//257 NA
256,
//255 NA
//254 NA
253,
//252 NA
251,
//250 NA
249,
248,
247,
//246 NA
245,
//244,
243,
242,
241,
240,
239,
//238,
237,
236,
235,
234,
233,
232,
231,
230,
229,
//228 NA
//227 NA
226,
//225 NA
//224 NA
//223 NA
//222 NA
221,
220,
219,
218,
//217 NA
//216 NA
215,
//214 NA
213,
//212 NA
211,
210,
209,
//208 NA
207,
//206 NA
205,
204,
203,
//202 NA
201,
//200 NA
199,
//198 NA
//197 NA
//196 NA
//195 NA
//194 NA
193,
192,
191,
//190 NA
//189 NA
//188 NA
187,
186,
//185 NA
184,
//183 NA
//182 NA
181,
//180 NA
//179 NA
178,
//177 NA
//176 NA
//175 NA
//174 NA
173,
172,
171,
170,
169,
//168 NA
167,
166,
165,
//164 NA
//163 NA
//162 NA
//161 NA
160,
159,
158,
157,
156,
155,
154,
153,
152,
151,
150,
149,
148,
147,
146,
145,
144,
143,
142,
141,
140,
139,
138,
137,
136,
135,
134,
133,
132,
131,
130,
129,
128,
127,
126,
125,
124,
123,
122,
121,
120,
119,
118,
117,
116,
115,
114,
113,
112,
111,
110,
109,
108,
107,
106,
105,
104,
103,
102,
101,
100,
99,
98,
97,
96,
95,
94,
93,
92,
91,
90,
89,
88,
87,
86,
85,
84,
83,
82,
81,
80,
79,
78,
77,
76,
75,
74,
73,
72,
71,
70,
69,
68,
67,
66,
65,
64,
63,
62,
61,
60,
59,
58,
57,
56,
55,
54,
53,
52,
51,
50,
49,
48,
47,
46,
45,
44,
43,
42,
41,
40,
39,
38,
37,
36,
35,
34,
33,
32,
31,
30,
29,
28,
27,
26,
25,
24,
23,
22,
21,
20,
19,
18,
17,
16,
15,
14,
13,
12,
11,
10,
9,
8,
7,
6,
5,
4,
3,
2,
1,
0
};
// clang-format on
/// Place to put a short description when adding a feature with a patch.
/// Keep it short, e.g.,: "relative numbers", "persistent undo".
/// Also add a comment marker to separate the lines.
/// See the official Vim patches for the diff format: It must use a context of
/// one line only. Create it by hand or use "diff -C2" and edit the patch.
static char *(extra_patches[]) = {
// Add your patch description below this line
NULL
};
/// Checks whether patch `n` has been included.
///
/// @param n The patch number.
///
/// @return TRUE if patch "n" has been included.
int has_patch(int n)
{
int i;
for (i = 0; included_patches[i] != 0; ++i) {
if (included_patches[i] == n) {
return TRUE;
}
}
return FALSE;
}
void ex_version(exarg_T *eap)
{
// Ignore a ":version 9.99" command.
if (*eap->arg == NUL) {
msg_putchar('\n');
list_version();
}
}
/// List all features aligned in columns, dictionary style.
static void list_features(void)
{
int nfeat = 0;
int width = 0;
// Find the length of the longest feature name, use that + 1 as the column
// width
int i;
for (i = 0; features[i] != NULL; ++i) {
int l = (int)STRLEN(features[i]);
if (l > width) {
width = l;
}
nfeat++;
}
width += 1;
if (Columns < width) {
// Not enough screen columns - show one per line
for (i = 0; features[i] != NULL; ++i) {
version_msg(features[i]);
if (msg_col > 0) {
msg_putchar('\n');
}
}
return;
}
// The rightmost column doesn't need a separator.
// Sacrifice it to fit in one more column if possible.
int ncol = (int)(Columns + 1) / width;
int nrow = nfeat / ncol + (nfeat % ncol ? 1 : 0);
// i counts columns then rows. idx counts rows then columns.
for (i = 0; !got_int && i < nrow * ncol; ++i) {
int idx = (i / ncol) + (i % ncol) * nrow;
if (idx < nfeat) {
int last_col = (i + 1) % ncol == 0;
msg_puts((char_u *)features[idx]);
if (last_col) {
if (msg_col > 0) {
msg_putchar('\n');
}
} else {
while (msg_col % width) {
msg_putchar(' ');
}
}
} else {
if (msg_col > 0) {
msg_putchar('\n');
}
}
}
MSG_PUTS("For differences from Vim, see :help vim-differences\n\n");
}
void list_version(void)
{
// When adding features here, don't forget to update the list of
// internal variables in eval.c!
MSG(longVersionWithDate);
MSG(version_commit);
MSG(version_buildtype);
MSG(version_cflags);
// Print the list of extra patch descriptions if there is at least one.
char *s = "";
if (extra_patches[0] != NULL) {
MSG_PUTS(_("\nExtra patches: "));
s = "";
for (int i = 0; extra_patches[i] != NULL; ++i) {
MSG_PUTS(s);
s = ", ";
MSG_PUTS(extra_patches[i]);
}
}
#ifdef HAVE_PATHDEF
if ((*compiled_user != NUL) || (*compiled_sys != NUL)) {
MSG_PUTS(_("\nCompiled "));
if (*compiled_user != NUL) {
MSG_PUTS(_("by "));
MSG_PUTS(compiled_user);
}
if (*compiled_sys != NUL) {
MSG_PUTS("@");
MSG_PUTS(compiled_sys);
}
}
#endif // ifdef HAVE_PATHDEF
version_msg(_("\n\nOptional features included (+) or not (-): "));
list_features();
#ifdef SYS_VIMRC_FILE
version_msg(_(" system vimrc file: \""));
version_msg(SYS_VIMRC_FILE);
version_msg("\"\n");
#endif // ifdef SYS_VIMRC_FILE
#ifdef USR_VIMRC_FILE
version_msg(_(" user vimrc file: \""));
version_msg(USR_VIMRC_FILE);
version_msg("\"\n");
#endif // ifdef USR_VIMRC_FILE
#ifdef USR_VIMRC_FILE2
version_msg(_(" 2nd user vimrc file: \""));
version_msg(USR_VIMRC_FILE2);
version_msg("\"\n");
#endif // ifdef USR_VIMRC_FILE2
#ifdef USR_VIMRC_FILE3
version_msg(_(" 3rd user vimrc file: \""));
version_msg(USR_VIMRC_FILE3);
version_msg("\"\n");
#endif // ifdef USR_VIMRC_FILE3
#ifdef USR_EXRC_FILE
version_msg(_(" user exrc file: \""));
version_msg(USR_EXRC_FILE);
version_msg("\"\n");
#endif // ifdef USR_EXRC_FILE
#ifdef USR_EXRC_FILE2
version_msg(_(" 2nd user exrc file: \""));
version_msg(USR_EXRC_FILE2);
version_msg("\"\n");
#endif // ifdef USR_EXRC_FILE2
#ifdef HAVE_PATHDEF
if (*default_vim_dir != NUL) {
version_msg(_(" fall-back for $VIM: \""));
version_msg(default_vim_dir);
version_msg("\"\n");
}
if (*default_vimruntime_dir != NUL) {
version_msg(_(" f-b for $VIMRUNTIME: \""));
version_msg(default_vimruntime_dir);
version_msg("\"\n");
}
#endif // ifdef HAVE_PATHDEF
}
/// Output a string for the version message. If it's going to wrap, output a
/// newline, unless the message is too long to fit on the screen anyway.
///
/// @param s
static void version_msg(char *s)
{
int len = (int)STRLEN(s);
if (!got_int
&& (len < (int)Columns)
&& (msg_col + len >= (int)Columns)
&& (*s != '\n')) {
msg_putchar('\n');
}
if (!got_int) {
MSG_PUTS(s);
}
}
/// Show the intro message when not editing a file.
void maybe_intro_message(void)
{
if (bufempty()
&& (curbuf->b_fname == NULL)
&& (firstwin->w_next == NULL)
&& (vim_strchr(p_shm, SHM_INTRO) == NULL)) {
intro_message(FALSE);
}
}
/// Give an introductory message about Vim.
/// Only used when starting Vim on an empty file, without a file name.
/// Or with the ":intro" command (for Sven :-).
///
/// @param colon TRUE for ":intro"
void intro_message(int colon)
{
int i;
long row;
long blanklines;
int sponsor;
char *p;
static char *(lines[]) = {
N_(NVIM_VERSION_LONG),
"",
N_("by Bram Moolenaar et al."),
N_("Vim is open source and freely distributable"),
"",
N_("Help poor children in Uganda!"),
N_("type :help iccf<Enter> for information "),
"",
N_("type :q<Enter> to exit "),
N_("type :help<Enter> or <F1> for on-line help"),
N_("type :help nvim<Enter> for Neovim help "),
};
// blanklines = screen height - # message lines
size_t lines_size = ARRAY_SIZE(lines);
assert(lines_size <= LONG_MAX);
blanklines = Rows - ((long)lines_size - 1l);
// Don't overwrite a statusline. Depends on 'cmdheight'.
if (p_ls > 1) {
blanklines -= Rows - topframe->fr_height;
}
if (blanklines < 0) {
blanklines = 0;
}
// Show the sponsor and register message one out of four times, the Uganda
// message two out of four times.
sponsor = (int)time(NULL);
sponsor = ((sponsor & 2) == 0) - ((sponsor & 4) == 0);
// start displaying the message lines after half of the blank lines
row = blanklines / 2;
if (((row >= 2) && (Columns >= 50)) || colon) {
for (i = 0; i < (int)ARRAY_SIZE(lines); ++i) {
p = lines[i];
if (sponsor != 0) {
if (strstr(p, "children") != NULL) {
p = sponsor < 0
? N_("Sponsor Vim development!")
: N_("Become a registered Vim user!");
} else if (strstr(p, "iccf") != NULL) {
p = sponsor < 0
? N_("type :help sponsor<Enter> for information ")
: N_("type :help register<Enter> for information ");
} else if (strstr(p, "Orphans") != NULL) {
p = N_("menu Help->Sponsor/Register for information ");
}
}
if (*p != NUL) {
do_intro_line(row, (char_u *)_(p), 0);
}
row++;
}
}
// Make the wait-return message appear just below the text.
if (colon) {
assert(row <= INT_MAX);
msg_row = (int)row;
}
}
static void do_intro_line(long row, char_u *mesg, int attr)
{
long col;
char_u *p;
int l;
int clen;
// Center the message horizontally.
col = vim_strsize(mesg);
col = (Columns - col) / 2;
if (col < 0) {
col = 0;
}
// Split up in parts to highlight <> items differently.
for (p = mesg; *p != NUL; p += l) {
clen = 0;
for (l = 0; p[l] != NUL
&& (l == 0 || (p[l] != '<' && p[l - 1] != '>')); ++l) {
if (has_mbyte) {
clen += ptr2cells(p + l);
l += (*mb_ptr2len)(p + l) - 1;
} else {
clen += byte2cells(p[l]);
}
}
assert(row <= INT_MAX && col <= INT_MAX);
screen_puts_len(p, l, (int)row, (int)col, *p == '<' ? hl_attr(HLF_8) : attr);
col += clen;
}
}
/// ":intro": clear screen, display intro screen and wait for return.
///
/// @param eap
void ex_intro(exarg_T *eap)
{
screenclear();
intro_message(TRUE);
wait_return(TRUE);
}