Mercurial > vim
annotate src/undo.c @ 2223:81b83a19e127 vim73
More strict checks for the undo file.
author | Bram Moolenaar <bram@vim.org> |
---|---|
date | Wed, 26 May 2010 21:21:00 +0200 |
parents | 271a5907f944 |
children | d45902a5c61c |
rev | line source |
---|---|
7 | 1 /* vi:set ts=8 sts=4 sw=4: |
2 * | |
3 * VIM - Vi IMproved by Bram Moolenaar | |
4 * | |
5 * Do ":help uganda" in Vim to read copying and usage conditions. | |
6 * Do ":help credits" in Vim to see a list of people who contributed. | |
7 * See README.txt for an overview of the Vim source code. | |
8 */ | |
9 | |
10 /* | |
11 * undo.c: multi level undo facility | |
12 * | |
13 * The saved lines are stored in a list of lists (one for each buffer): | |
14 * | |
15 * b_u_oldhead------------------------------------------------+ | |
16 * | | |
17 * V | |
18 * +--------------+ +--------------+ +--------------+ | |
19 * b_u_newhead--->| u_header | | u_header | | u_header | | |
20 * | uh_next------>| uh_next------>| uh_next---->NULL | |
21 * NULL<--------uh_prev |<---------uh_prev |<---------uh_prev | | |
22 * | uh_entry | | uh_entry | | uh_entry | | |
23 * +--------|-----+ +--------|-----+ +--------|-----+ | |
24 * | | | | |
25 * V V V | |
26 * +--------------+ +--------------+ +--------------+ | |
27 * | u_entry | | u_entry | | u_entry | | |
28 * | ue_next | | ue_next | | ue_next | | |
29 * +--------|-----+ +--------|-----+ +--------|-----+ | |
30 * | | | | |
31 * V V V | |
32 * +--------------+ NULL NULL | |
33 * | u_entry | | |
34 * | ue_next | | |
35 * +--------|-----+ | |
36 * | | |
37 * V | |
38 * etc. | |
39 * | |
40 * Each u_entry list contains the information for one undo or redo. | |
41 * curbuf->b_u_curhead points to the header of the last undo (the next redo), | |
758 | 42 * or is NULL if nothing has been undone (end of the branch). |
7 | 43 * |
753 | 44 * For keeping alternate undo/redo branches the uh_alt field is used. Thus at |
45 * each point in the list a branch may appear for an alternate to redo. The | |
46 * uh_seq field is numbered sequentially to be able to find a newer or older | |
47 * branch. | |
48 * | |
758 | 49 * +---------------+ +---------------+ |
50 * b_u_oldhead --->| u_header | | u_header | | |
51 * | uh_alt_next ---->| uh_alt_next ----> NULL | |
52 * NULL <----- uh_alt_prev |<------ uh_alt_prev | | |
53 * | uh_prev | | uh_prev | | |
54 * +-----|---------+ +-----|---------+ | |
55 * | | | |
56 * V V | |
57 * +---------------+ +---------------+ | |
58 * | u_header | | u_header | | |
59 * | uh_alt_next | | uh_alt_next | | |
60 * b_u_newhead --->| uh_alt_prev | | uh_alt_prev | | |
61 * | uh_prev | | uh_prev | | |
62 * +-----|---------+ +-----|---------+ | |
63 * | | | |
64 * V V | |
65 * NULL +---------------+ +---------------+ | |
66 * | u_header | | u_header | | |
67 * | uh_alt_next ---->| uh_alt_next | | |
68 * | uh_alt_prev |<------ uh_alt_prev | | |
69 * | uh_prev | | uh_prev | | |
70 * +-----|---------+ +-----|---------+ | |
71 * | | | |
72 * etc. etc. | |
73 * | |
74 * | |
168 | 75 * All data is allocated with U_ALLOC_LINE(), it will be freed as soon as the |
76 * buffer is unloaded. | |
7 | 77 */ |
78 | |
1415 | 79 /* Uncomment the next line for including the u_check() function. This warns |
80 * for errors in the debug information. */ | |
81 /* #define U_DEBUG 1 */ | |
82 #define UH_MAGIC 0x18dade /* value for uh_magic when in use */ | |
83 #define UE_MAGIC 0xabc123 /* value for ue_magic when in use */ | |
84 | |
2217
120502692d82
Improve the MS-Windows installer.
Bram Moolenaar <bram@vim.org>
parents:
2215
diff
changeset
|
85 #if defined(MSDOS) || defined(WIN16) || defined(WIN32) || defined(_WIN64) |
120502692d82
Improve the MS-Windows installer.
Bram Moolenaar <bram@vim.org>
parents:
2215
diff
changeset
|
86 # include "vimio.h" /* for vim_read(), must be before vim.h */ |
120502692d82
Improve the MS-Windows installer.
Bram Moolenaar <bram@vim.org>
parents:
2215
diff
changeset
|
87 #endif |
120502692d82
Improve the MS-Windows installer.
Bram Moolenaar <bram@vim.org>
parents:
2215
diff
changeset
|
88 |
7 | 89 #include "vim.h" |
90 | |
168 | 91 /* See below: use malloc()/free() for memory management. */ |
92 #define U_USE_MALLOC 1 | |
93 | |
758 | 94 static void u_unch_branch __ARGS((u_header_T *uhp)); |
7 | 95 static u_entry_T *u_get_headentry __ARGS((void)); |
96 static void u_getbot __ARGS((void)); | |
97 static int u_savecommon __ARGS((linenr_T, linenr_T, linenr_T)); | |
98 static void u_doit __ARGS((int count)); | |
777 | 99 static void u_undoredo __ARGS((int undo)); |
798 | 100 static void u_undo_end __ARGS((int did_undo, int absolute)); |
772 | 101 static void u_add_time __ARGS((char_u *buf, size_t buflen, time_t tt)); |
758 | 102 static void u_freeheader __ARGS((buf_T *buf, u_header_T *uhp, u_header_T **uhpp)); |
753 | 103 static void u_freebranch __ARGS((buf_T *buf, u_header_T *uhp, u_header_T **uhpp)); |
104 static void u_freeentries __ARGS((buf_T *buf, u_header_T *uhp, u_header_T **uhpp)); | |
7 | 105 static void u_freeentry __ARGS((u_entry_T *, long)); |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
106 #ifdef FEAT_PERSISTENT_UNDO |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
107 static void unserialize_pos __ARGS((pos_T *pos, FILE *fp)); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
108 static void unserialize_visualinfo __ARGS((visualinfo_T *info, FILE *fp)); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
109 static char_u *u_get_undo_file_name __ARGS((char_u *, int reading)); |
2223
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
110 static void u_free_uhp __ARGS((u_header_T *uhp)); |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
111 static int serialize_uep __ARGS((u_entry_T *uep, FILE *fp)); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
112 static void serialize_pos __ARGS((pos_T pos, FILE *fp)); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
113 static void serialize_visualinfo __ARGS((visualinfo_T info, FILE *fp)); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
114 #endif |
7 | 115 |
168 | 116 #ifdef U_USE_MALLOC |
117 # define U_FREE_LINE(ptr) vim_free(ptr) | |
118 # define U_ALLOC_LINE(size) lalloc((long_u)((size) + 1), FALSE) | |
119 #else | |
120 static void u_free_line __ARGS((char_u *ptr, int keep)); | |
121 static char_u *u_alloc_line __ARGS((unsigned size)); | |
122 # define U_FREE_LINE(ptr) u_free_line((ptr), FALSE) | |
123 # define U_ALLOC_LINE(size) u_alloc_line(size) | |
124 #endif | |
7 | 125 static char_u *u_save_line __ARGS((linenr_T)); |
126 | |
127 static long u_newcount, u_oldcount; | |
128 | |
129 /* | |
130 * When 'u' flag included in 'cpoptions', we behave like vi. Need to remember | |
131 * the action that "u" should do. | |
132 */ | |
133 static int undo_undoes = FALSE; | |
134 | |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
135 static int lastmark = 0; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
136 |
1415 | 137 #ifdef U_DEBUG |
138 /* | |
139 * Check the undo structures for being valid. Print a warning when something | |
140 * looks wrong. | |
141 */ | |
142 static int seen_b_u_curhead; | |
143 static int seen_b_u_newhead; | |
144 static int header_count; | |
145 | |
146 static void | |
147 u_check_tree(u_header_T *uhp, | |
148 u_header_T *exp_uh_next, | |
149 u_header_T *exp_uh_alt_prev) | |
150 { | |
151 u_entry_T *uep; | |
152 | |
153 if (uhp == NULL) | |
154 return; | |
155 ++header_count; | |
156 if (uhp == curbuf->b_u_curhead && ++seen_b_u_curhead > 1) | |
157 { | |
158 EMSG("b_u_curhead found twice (looping?)"); | |
159 return; | |
160 } | |
161 if (uhp == curbuf->b_u_newhead && ++seen_b_u_newhead > 1) | |
162 { | |
163 EMSG("b_u_newhead found twice (looping?)"); | |
164 return; | |
165 } | |
166 | |
167 if (uhp->uh_magic != UH_MAGIC) | |
168 EMSG("uh_magic wrong (may be using freed memory)"); | |
169 else | |
170 { | |
171 /* Check pointers back are correct. */ | |
172 if (uhp->uh_next != exp_uh_next) | |
173 { | |
174 EMSG("uh_next wrong"); | |
175 smsg((char_u *)"expected: 0x%x, actual: 0x%x", | |
176 exp_uh_next, uhp->uh_next); | |
177 } | |
178 if (uhp->uh_alt_prev != exp_uh_alt_prev) | |
179 { | |
180 EMSG("uh_alt_prev wrong"); | |
181 smsg((char_u *)"expected: 0x%x, actual: 0x%x", | |
182 exp_uh_alt_prev, uhp->uh_alt_prev); | |
183 } | |
184 | |
185 /* Check the undo tree at this header. */ | |
186 for (uep = uhp->uh_entry; uep != NULL; uep = uep->ue_next) | |
187 { | |
188 if (uep->ue_magic != UE_MAGIC) | |
189 { | |
190 EMSG("ue_magic wrong (may be using freed memory)"); | |
191 break; | |
192 } | |
193 } | |
194 | |
195 /* Check the next alt tree. */ | |
196 u_check_tree(uhp->uh_alt_next, uhp->uh_next, uhp); | |
197 | |
198 /* Check the next header in this branch. */ | |
199 u_check_tree(uhp->uh_prev, uhp, NULL); | |
200 } | |
201 } | |
202 | |
203 void | |
204 u_check(int newhead_may_be_NULL) | |
205 { | |
206 seen_b_u_newhead = 0; | |
207 seen_b_u_curhead = 0; | |
208 header_count = 0; | |
209 | |
210 u_check_tree(curbuf->b_u_oldhead, NULL, NULL); | |
211 | |
212 if (seen_b_u_newhead == 0 && curbuf->b_u_oldhead != NULL | |
213 && !(newhead_may_be_NULL && curbuf->b_u_newhead == NULL)) | |
214 EMSGN("b_u_newhead invalid: 0x%x", curbuf->b_u_newhead); | |
215 if (curbuf->b_u_curhead != NULL && seen_b_u_curhead == 0) | |
216 EMSGN("b_u_curhead invalid: 0x%x", curbuf->b_u_curhead); | |
217 if (header_count != curbuf->b_u_numhead) | |
218 { | |
219 EMSG("b_u_numhead invalid"); | |
220 smsg((char_u *)"expected: %ld, actual: %ld", | |
221 (long)header_count, (long)curbuf->b_u_numhead); | |
222 } | |
223 } | |
224 #endif | |
225 | |
7 | 226 /* |
344 | 227 * Save the current line for both the "u" and "U" command. |
228 * Returns OK or FAIL. | |
7 | 229 */ |
230 int | |
231 u_save_cursor() | |
232 { | |
233 return (u_save((linenr_T)(curwin->w_cursor.lnum - 1), | |
234 (linenr_T)(curwin->w_cursor.lnum + 1))); | |
235 } | |
236 | |
237 /* | |
238 * Save the lines between "top" and "bot" for both the "u" and "U" command. | |
239 * "top" may be 0 and bot may be curbuf->b_ml.ml_line_count + 1. | |
240 * Returns FAIL when lines could not be saved, OK otherwise. | |
241 */ | |
242 int | |
243 u_save(top, bot) | |
244 linenr_T top, bot; | |
245 { | |
246 if (undo_off) | |
247 return OK; | |
248 | |
249 if (top > curbuf->b_ml.ml_line_count || | |
250 top >= bot || bot > curbuf->b_ml.ml_line_count + 1) | |
251 return FALSE; /* rely on caller to do error messages */ | |
252 | |
253 if (top + 2 == bot) | |
254 u_saveline((linenr_T)(top + 1)); | |
255 | |
256 return (u_savecommon(top, bot, (linenr_T)0)); | |
257 } | |
258 | |
259 /* | |
2154
7c8c7c95a865
First step in the Vim 7.3 branch. Changed version numbers.
Bram Moolenaar <bram@zimbu.org>
parents:
1534
diff
changeset
|
260 * Save the line "lnum" (used by ":s" and "~" command). |
7 | 261 * The line is replaced, so the new bottom line is lnum + 1. |
262 */ | |
263 int | |
264 u_savesub(lnum) | |
265 linenr_T lnum; | |
266 { | |
267 if (undo_off) | |
268 return OK; | |
269 | |
270 return (u_savecommon(lnum - 1, lnum + 1, lnum + 1)); | |
271 } | |
272 | |
273 /* | |
2154
7c8c7c95a865
First step in the Vim 7.3 branch. Changed version numbers.
Bram Moolenaar <bram@zimbu.org>
parents:
1534
diff
changeset
|
274 * A new line is inserted before line "lnum" (used by :s command). |
7 | 275 * The line is inserted, so the new bottom line is lnum + 1. |
276 */ | |
277 int | |
278 u_inssub(lnum) | |
279 linenr_T lnum; | |
280 { | |
281 if (undo_off) | |
282 return OK; | |
283 | |
284 return (u_savecommon(lnum - 1, lnum, lnum + 1)); | |
285 } | |
286 | |
287 /* | |
2154
7c8c7c95a865
First step in the Vim 7.3 branch. Changed version numbers.
Bram Moolenaar <bram@zimbu.org>
parents:
1534
diff
changeset
|
288 * Save the lines "lnum" - "lnum" + nlines (used by delete command). |
7 | 289 * The lines are deleted, so the new bottom line is lnum, unless the buffer |
290 * becomes empty. | |
291 */ | |
292 int | |
293 u_savedel(lnum, nlines) | |
294 linenr_T lnum; | |
295 long nlines; | |
296 { | |
297 if (undo_off) | |
298 return OK; | |
299 | |
300 return (u_savecommon(lnum - 1, lnum + nlines, | |
301 nlines == curbuf->b_ml.ml_line_count ? 2 : lnum)); | |
302 } | |
303 | |
632 | 304 /* |
305 * Return TRUE when undo is allowed. Otherwise give an error message and | |
306 * return FALSE. | |
307 */ | |
912 | 308 int |
632 | 309 undo_allowed() |
310 { | |
311 /* Don't allow changes when 'modifiable' is off. */ | |
312 if (!curbuf->b_p_ma) | |
313 { | |
314 EMSG(_(e_modifiable)); | |
315 return FALSE; | |
316 } | |
317 | |
318 #ifdef HAVE_SANDBOX | |
319 /* In the sandbox it's not allowed to change the text. */ | |
320 if (sandbox != 0) | |
321 { | |
322 EMSG(_(e_sandbox)); | |
323 return FALSE; | |
324 } | |
325 #endif | |
326 | |
327 /* Don't allow changes in the buffer while editing the cmdline. The | |
328 * caller of getcmdline() may get confused. */ | |
634 | 329 if (textlock != 0) |
632 | 330 { |
331 EMSG(_(e_secure)); | |
332 return FALSE; | |
333 } | |
334 | |
335 return TRUE; | |
336 } | |
337 | |
7 | 338 static int |
339 u_savecommon(top, bot, newbot) | |
340 linenr_T top, bot; | |
341 linenr_T newbot; | |
342 { | |
753 | 343 linenr_T lnum; |
344 long i; | |
345 u_header_T *uhp; | |
346 u_header_T *old_curhead; | |
347 u_entry_T *uep; | |
348 u_entry_T *prev_uep; | |
349 long size; | |
7 | 350 |
632 | 351 /* When making changes is not allowed return FAIL. It's a crude way to |
352 * make all change commands fail. */ | |
353 if (!undo_allowed()) | |
7 | 354 return FAIL; |
355 | |
1415 | 356 #ifdef U_DEBUG |
357 u_check(FALSE); | |
358 #endif | |
7 | 359 #ifdef FEAT_NETBEANS_INTG |
360 /* | |
361 * Netbeans defines areas that cannot be modified. Bail out here when | |
362 * trying to change text in a guarded area. | |
363 */ | |
2210 | 364 if (netbeans_active()) |
7 | 365 { |
33 | 366 if (netbeans_is_guarded(top, bot)) |
367 { | |
368 EMSG(_(e_guarded)); | |
369 return FAIL; | |
370 } | |
371 if (curbuf->b_p_ro) | |
372 { | |
373 EMSG(_(e_nbreadonly)); | |
374 return FAIL; | |
375 } | |
7 | 376 } |
377 #endif | |
378 | |
379 #ifdef FEAT_AUTOCMD | |
380 /* | |
381 * Saving text for undo means we are going to make a change. Give a | |
382 * warning for a read-only file before making the change, so that the | |
383 * FileChangedRO event can replace the buffer with a read-write version | |
384 * (e.g., obtained from a source control system). | |
385 */ | |
386 change_warning(0); | |
387 #endif | |
388 | |
389 size = bot - top - 1; | |
390 | |
391 /* | |
392 * if curbuf->b_u_synced == TRUE make a new header | |
393 */ | |
394 if (curbuf->b_u_synced) | |
395 { | |
396 #ifdef FEAT_JUMPLIST | |
397 /* Need to create new entry in b_changelist. */ | |
398 curbuf->b_new_change = TRUE; | |
399 #endif | |
400 | |
753 | 401 if (p_ul >= 0) |
402 { | |
403 /* | |
404 * Make a new header entry. Do this first so that we don't mess | |
405 * up the undo info when out of memory. | |
406 */ | |
407 uhp = (u_header_T *)U_ALLOC_LINE((unsigned)sizeof(u_header_T)); | |
408 if (uhp == NULL) | |
409 goto nomem; | |
1415 | 410 #ifdef U_DEBUG |
411 uhp->uh_magic = UH_MAGIC; | |
412 #endif | |
753 | 413 } |
766 | 414 else |
415 uhp = NULL; | |
753 | 416 |
7 | 417 /* |
758 | 418 * If we undid more than we redid, move the entry lists before and |
419 * including curbuf->b_u_curhead to an alternate branch. | |
7 | 420 */ |
753 | 421 old_curhead = curbuf->b_u_curhead; |
422 if (old_curhead != NULL) | |
423 { | |
424 curbuf->b_u_newhead = old_curhead->uh_next; | |
425 curbuf->b_u_curhead = NULL; | |
426 } | |
7 | 427 |
428 /* | |
429 * free headers to keep the size right | |
430 */ | |
431 while (curbuf->b_u_numhead > p_ul && curbuf->b_u_oldhead != NULL) | |
753 | 432 { |
758 | 433 u_header_T *uhfree = curbuf->b_u_oldhead; |
753 | 434 |
1415 | 435 if (uhfree == old_curhead) |
436 /* Can't reconnect the branch, delete all of it. */ | |
437 u_freebranch(curbuf, uhfree, &old_curhead); | |
438 else if (uhfree->uh_alt_next == NULL) | |
439 /* There is no branch, only free one header. */ | |
758 | 440 u_freeheader(curbuf, uhfree, &old_curhead); |
753 | 441 else |
442 { | |
443 /* Free the oldest alternate branch as a whole. */ | |
758 | 444 while (uhfree->uh_alt_next != NULL) |
445 uhfree = uhfree->uh_alt_next; | |
446 u_freebranch(curbuf, uhfree, &old_curhead); | |
753 | 447 } |
1415 | 448 #ifdef U_DEBUG |
449 u_check(TRUE); | |
450 #endif | |
753 | 451 } |
7 | 452 |
766 | 453 if (uhp == NULL) /* no undo at all */ |
7 | 454 { |
753 | 455 if (old_curhead != NULL) |
456 u_freebranch(curbuf, old_curhead, NULL); | |
7 | 457 curbuf->b_u_synced = FALSE; |
458 return OK; | |
459 } | |
460 | |
461 uhp->uh_prev = NULL; | |
462 uhp->uh_next = curbuf->b_u_newhead; | |
753 | 463 uhp->uh_alt_next = old_curhead; |
464 if (old_curhead != NULL) | |
465 { | |
1056 | 466 uhp->uh_alt_prev = old_curhead->uh_alt_prev; |
467 if (uhp->uh_alt_prev != NULL) | |
468 uhp->uh_alt_prev->uh_alt_next = uhp; | |
753 | 469 old_curhead->uh_alt_prev = uhp; |
470 if (curbuf->b_u_oldhead == old_curhead) | |
471 curbuf->b_u_oldhead = uhp; | |
472 } | |
1056 | 473 else |
474 uhp->uh_alt_prev = NULL; | |
758 | 475 if (curbuf->b_u_newhead != NULL) |
476 curbuf->b_u_newhead->uh_prev = uhp; | |
477 | |
777 | 478 uhp->uh_seq = ++curbuf->b_u_seq_last; |
479 curbuf->b_u_seq_cur = uhp->uh_seq; | |
758 | 480 uhp->uh_time = time(NULL); |
481 curbuf->b_u_seq_time = uhp->uh_time + 1; | |
482 | |
753 | 483 uhp->uh_walk = 0; |
7 | 484 uhp->uh_entry = NULL; |
485 uhp->uh_getbot_entry = NULL; | |
486 uhp->uh_cursor = curwin->w_cursor; /* save cursor pos. for undo */ | |
487 #ifdef FEAT_VIRTUALEDIT | |
488 if (virtual_active() && curwin->w_cursor.coladd > 0) | |
489 uhp->uh_cursor_vcol = getviscol(); | |
490 else | |
491 uhp->uh_cursor_vcol = -1; | |
492 #endif | |
493 | |
494 /* save changed and buffer empty flag for undo */ | |
495 uhp->uh_flags = (curbuf->b_changed ? UH_CHANGED : 0) + | |
496 ((curbuf->b_ml.ml_flags & ML_EMPTY) ? UH_EMPTYBUF : 0); | |
497 | |
692 | 498 /* save named marks and Visual marks for undo */ |
7 | 499 mch_memmove(uhp->uh_namedm, curbuf->b_namedm, sizeof(pos_T) * NMARKS); |
692 | 500 #ifdef FEAT_VISUAL |
501 uhp->uh_visual = curbuf->b_visual; | |
502 #endif | |
503 | |
7 | 504 curbuf->b_u_newhead = uhp; |
505 if (curbuf->b_u_oldhead == NULL) | |
506 curbuf->b_u_oldhead = uhp; | |
507 ++curbuf->b_u_numhead; | |
508 } | |
509 else | |
510 { | |
511 if (p_ul < 0) /* no undo at all */ | |
512 return OK; | |
513 | |
514 /* | |
515 * When saving a single line, and it has been saved just before, it | |
516 * doesn't make sense saving it again. Saves a lot of memory when | |
517 * making lots of changes inside the same line. | |
518 * This is only possible if the previous change didn't increase or | |
519 * decrease the number of lines. | |
520 * Check the ten last changes. More doesn't make sense and takes too | |
521 * long. | |
522 */ | |
523 if (size == 1) | |
524 { | |
525 uep = u_get_headentry(); | |
526 prev_uep = NULL; | |
527 for (i = 0; i < 10; ++i) | |
528 { | |
529 if (uep == NULL) | |
530 break; | |
531 | |
532 /* If lines have been inserted/deleted we give up. | |
533 * Also when the line was included in a multi-line save. */ | |
534 if ((curbuf->b_u_newhead->uh_getbot_entry != uep | |
535 ? (uep->ue_top + uep->ue_size + 1 | |
536 != (uep->ue_bot == 0 | |
537 ? curbuf->b_ml.ml_line_count + 1 | |
538 : uep->ue_bot)) | |
539 : uep->ue_lcount != curbuf->b_ml.ml_line_count) | |
540 || (uep->ue_size > 1 | |
541 && top >= uep->ue_top | |
542 && top + 2 <= uep->ue_top + uep->ue_size + 1)) | |
543 break; | |
544 | |
545 /* If it's the same line we can skip saving it again. */ | |
546 if (uep->ue_size == 1 && uep->ue_top == top) | |
547 { | |
548 if (i > 0) | |
549 { | |
550 /* It's not the last entry: get ue_bot for the last | |
551 * entry now. Following deleted/inserted lines go to | |
552 * the re-used entry. */ | |
553 u_getbot(); | |
554 curbuf->b_u_synced = FALSE; | |
555 | |
556 /* Move the found entry to become the last entry. The | |
557 * order of undo/redo doesn't matter for the entries | |
558 * we move it over, since they don't change the line | |
559 * count and don't include this line. It does matter | |
560 * for the found entry if the line count is changed by | |
561 * the executed command. */ | |
562 prev_uep->ue_next = uep->ue_next; | |
563 uep->ue_next = curbuf->b_u_newhead->uh_entry; | |
564 curbuf->b_u_newhead->uh_entry = uep; | |
565 } | |
566 | |
567 /* The executed command may change the line count. */ | |
568 if (newbot != 0) | |
569 uep->ue_bot = newbot; | |
570 else if (bot > curbuf->b_ml.ml_line_count) | |
571 uep->ue_bot = 0; | |
572 else | |
573 { | |
574 uep->ue_lcount = curbuf->b_ml.ml_line_count; | |
575 curbuf->b_u_newhead->uh_getbot_entry = uep; | |
576 } | |
577 return OK; | |
578 } | |
579 prev_uep = uep; | |
580 uep = uep->ue_next; | |
581 } | |
582 } | |
583 | |
584 /* find line number for ue_bot for previous u_save() */ | |
585 u_getbot(); | |
586 } | |
587 | |
588 #if !defined(UNIX) && !defined(DJGPP) && !defined(WIN32) && !defined(__EMX__) | |
589 /* | |
590 * With Amiga and MSDOS 16 bit we can't handle big undo's, because | |
591 * then u_alloc_line would have to allocate a block larger than 32K | |
592 */ | |
593 if (size >= 8000) | |
594 goto nomem; | |
595 #endif | |
596 | |
597 /* | |
598 * add lines in front of entry list | |
599 */ | |
168 | 600 uep = (u_entry_T *)U_ALLOC_LINE((unsigned)sizeof(u_entry_T)); |
7 | 601 if (uep == NULL) |
602 goto nomem; | |
2215
cccb71c2c5c1
Fix uninit memory read in undo code. Fix uint32_t in proto file.
Bram Moolenaar <bram@vim.org>
parents:
2214
diff
changeset
|
603 vim_memset(uep, 0, sizeof(u_entry_T)); |
1415 | 604 #ifdef U_DEBUG |
605 uep->ue_magic = UE_MAGIC; | |
606 #endif | |
7 | 607 |
608 uep->ue_size = size; | |
609 uep->ue_top = top; | |
610 if (newbot != 0) | |
611 uep->ue_bot = newbot; | |
612 /* | |
613 * Use 0 for ue_bot if bot is below last line. | |
614 * Otherwise we have to compute ue_bot later. | |
615 */ | |
616 else if (bot > curbuf->b_ml.ml_line_count) | |
617 uep->ue_bot = 0; | |
618 else | |
619 { | |
620 uep->ue_lcount = curbuf->b_ml.ml_line_count; | |
621 curbuf->b_u_newhead->uh_getbot_entry = uep; | |
622 } | |
623 | |
168 | 624 if (size > 0) |
7 | 625 { |
168 | 626 if ((uep->ue_array = (char_u **)U_ALLOC_LINE( |
7 | 627 (unsigned)(sizeof(char_u *) * size))) == NULL) |
628 { | |
629 u_freeentry(uep, 0L); | |
630 goto nomem; | |
631 } | |
632 for (i = 0, lnum = top + 1; i < size; ++i) | |
633 { | |
481 | 634 fast_breakcheck(); |
635 if (got_int) | |
636 { | |
637 u_freeentry(uep, i); | |
638 return FAIL; | |
639 } | |
7 | 640 if ((uep->ue_array[i] = u_save_line(lnum++)) == NULL) |
641 { | |
642 u_freeentry(uep, i); | |
643 goto nomem; | |
644 } | |
645 } | |
646 } | |
359 | 647 else |
648 uep->ue_array = NULL; | |
7 | 649 uep->ue_next = curbuf->b_u_newhead->uh_entry; |
650 curbuf->b_u_newhead->uh_entry = uep; | |
651 curbuf->b_u_synced = FALSE; | |
652 undo_undoes = FALSE; | |
653 | |
1415 | 654 #ifdef U_DEBUG |
655 u_check(FALSE); | |
656 #endif | |
7 | 657 return OK; |
658 | |
659 nomem: | |
660 msg_silent = 0; /* must display the prompt */ | |
661 if (ask_yesno((char_u *)_("No undo possible; continue anyway"), TRUE) | |
662 == 'y') | |
663 { | |
664 undo_off = TRUE; /* will be reset when character typed */ | |
665 return OK; | |
666 } | |
667 do_outofmem_msg((long_u)0); | |
668 return FAIL; | |
669 } | |
670 | |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
671 #ifdef FEAT_PERSISTENT_UNDO |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
672 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
673 # define UF_START_MAGIC 0xfeac /* magic at start of undofile */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
674 # define UF_HEADER_MAGIC 0x5fd0 /* magic at start of header */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
675 # define UF_END_MAGIC 0xe7aa /* magic after last header */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
676 # define UF_VERSION 1 /* 2-byte undofile version number */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
677 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
678 /* |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
679 * Compute the hash for the current buffer text into hash[UNDO_HASH_SIZE]. |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
680 */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
681 void |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
682 u_compute_hash(hash) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
683 char_u *hash; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
684 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
685 context_sha256_T ctx; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
686 linenr_T lnum; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
687 char_u *p; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
688 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
689 sha256_start(&ctx); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
690 for (lnum = 1; lnum < curbuf->b_ml.ml_line_count; ++lnum) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
691 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
692 p = ml_get(lnum); |
2217
120502692d82
Improve the MS-Windows installer.
Bram Moolenaar <bram@vim.org>
parents:
2215
diff
changeset
|
693 sha256_update(&ctx, p, (UINT32_T)(STRLEN(p) + 1)); |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
694 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
695 sha256_finish(&ctx, hash); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
696 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
697 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
698 /* |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
699 * Unserialize the pos_T at the current position in fp. |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
700 */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
701 static void |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
702 unserialize_pos(pos, fp) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
703 pos_T *pos; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
704 FILE *fp; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
705 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
706 pos->lnum = get4c(fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
707 pos->col = get4c(fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
708 #ifdef FEAT_VIRTUALEDIT |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
709 pos->coladd = get4c(fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
710 #else |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
711 (void)get4c(fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
712 #endif |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
713 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
714 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
715 /* |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
716 * Unserialize the visualinfo_T at the current position in fp. |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
717 */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
718 static void |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
719 unserialize_visualinfo(info, fp) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
720 visualinfo_T *info; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
721 FILE *fp; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
722 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
723 unserialize_pos(&info->vi_start, fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
724 unserialize_pos(&info->vi_end, fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
725 info->vi_mode = get4c(fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
726 info->vi_curswant = get4c(fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
727 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
728 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
729 /* |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
730 * Return an allocated string of the full path of the target undofile. |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
731 * When "reading" is TRUE find the file to read, go over all directories in |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
732 * 'undodir'. |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
733 * When "reading" is FALSE use the first name where the directory exists. |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
734 */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
735 static char_u * |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
736 u_get_undo_file_name(buf_ffname, reading) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
737 char_u *buf_ffname; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
738 int reading; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
739 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
740 char_u *dirp; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
741 char_u dir_name[IOSIZE + 1]; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
742 char_u *munged_name = NULL; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
743 char_u *undo_file_name = NULL; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
744 int dir_len; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
745 char_u *p; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
746 struct stat st; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
747 char_u *ffname = buf_ffname; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
748 #ifdef HAVE_READLINK |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
749 char_u fname_buf[MAXPATHL]; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
750 #endif |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
751 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
752 if (ffname == NULL) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
753 return NULL; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
754 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
755 #ifdef HAVE_READLINK |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
756 /* Expand symlink in the file name, so that we put the undo file with the |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
757 * actual file instead of with the symlink. */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
758 if (resolve_symlink(ffname, fname_buf) == OK) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
759 ffname = fname_buf; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
760 #endif |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
761 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
762 /* Loop over 'undodir'. When reading find the first file that exists. |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
763 * When not reading use the first directory that exists or ".". */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
764 dirp = p_udir; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
765 while (*dirp != NUL) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
766 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
767 dir_len = copy_option_part(&dirp, dir_name, IOSIZE, ","); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
768 if (dir_len == 1 && dir_name[0] == '.') |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
769 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
770 /* Use same directory as the ffname, |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
771 * "dir/name" -> "dir/.name.un~" */ |
2217
120502692d82
Improve the MS-Windows installer.
Bram Moolenaar <bram@vim.org>
parents:
2215
diff
changeset
|
772 undo_file_name = vim_strnsave(ffname, (int)(STRLEN(ffname) + 5)); |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
773 if (undo_file_name == NULL) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
774 break; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
775 p = gettail(undo_file_name); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
776 mch_memmove(p + 1, p, STRLEN(p) + 1); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
777 *p = '.'; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
778 STRCAT(p, ".un~"); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
779 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
780 else |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
781 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
782 dir_name[dir_len] = NUL; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
783 if (mch_isdir(dir_name)) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
784 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
785 if (munged_name == NULL) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
786 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
787 munged_name = vim_strsave(ffname); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
788 if (munged_name == NULL) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
789 return NULL; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
790 for (p = munged_name; *p != NUL; mb_ptr_adv(p)) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
791 if (vim_ispathsep(*p)) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
792 *p = '%'; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
793 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
794 undo_file_name = concat_fnames(dir_name, munged_name, TRUE); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
795 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
796 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
797 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
798 /* When reading check if the file exists. */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
799 if (undo_file_name != NULL && (!reading |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
800 || mch_stat((char *)undo_file_name, &st) >= 0)) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
801 break; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
802 vim_free(undo_file_name); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
803 undo_file_name = NULL; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
804 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
805 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
806 vim_free(munged_name); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
807 return undo_file_name; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
808 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
809 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
810 /* |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
811 * Load the undo tree from an undo file. |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
812 * If "name" is not NULL use it as the undo file name. This also means being |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
813 * a bit more verbose. |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
814 * Otherwise use curbuf->b_ffname to generate the undo file name. |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
815 * "hash[UNDO_HASH_SIZE]" must be the hash value of the buffer text. |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
816 */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
817 void |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
818 u_read_undo(name, hash) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
819 char_u *name; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
820 char_u *hash; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
821 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
822 char_u *file_name; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
823 FILE *fp; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
824 long magic, version, str_len; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
825 char_u *line_ptr = NULL; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
826 linenr_T line_lnum; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
827 colnr_T line_colnr; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
828 linenr_T line_count; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
829 int uep_len; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
830 int line_len; |
2217
120502692d82
Improve the MS-Windows installer.
Bram Moolenaar <bram@vim.org>
parents:
2215
diff
changeset
|
831 int num_head = 0; |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
832 long old_header_seq, new_header_seq, cur_header_seq; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
833 long seq_last, seq_cur; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
834 short old_idx = -1, new_idx = -1, cur_idx = -1; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
835 long num_read_uhps = 0; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
836 time_t seq_time; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
837 int i, j; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
838 int c; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
839 char_u **array; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
840 char_u *line; |
2223
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
841 u_entry_T *uep, *last_uep; |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
842 u_header_T *uhp; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
843 u_header_T **uhp_table = NULL; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
844 char_u read_hash[UNDO_HASH_SIZE]; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
845 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
846 if (name == NULL) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
847 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
848 file_name = u_get_undo_file_name(curbuf->b_ffname, TRUE); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
849 if (file_name == NULL) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
850 return; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
851 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
852 else |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
853 file_name = name; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
854 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
855 if (p_verbose > 0) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
856 smsg((char_u *)_("Reading undo file: %s"), file_name); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
857 fp = mch_fopen((char *)file_name, "r"); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
858 if (fp == NULL) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
859 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
860 if (name != NULL || p_verbose > 0) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
861 EMSG2(_("E822: Cannot open undo file for reading: %s"), file_name); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
862 goto error; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
863 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
864 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
865 /* Begin overall file information */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
866 magic = get2c(fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
867 if (magic != UF_START_MAGIC) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
868 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
869 EMSG2(_("E823: Corrupted undo file: %s"), file_name); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
870 goto error; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
871 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
872 version = get2c(fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
873 if (version != UF_VERSION) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
874 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
875 EMSG2(_("E824: Incompatible undo file: %s"), file_name); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
876 goto error; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
877 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
878 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
879 fread(read_hash, UNDO_HASH_SIZE, 1, fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
880 line_count = (linenr_T)get4c(fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
881 if (memcmp(hash, read_hash, UNDO_HASH_SIZE) != 0 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
882 || line_count != curbuf->b_ml.ml_line_count) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
883 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
884 if (p_verbose > 0 || name != NULL) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
885 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
886 verbose_enter(); |
2215
cccb71c2c5c1
Fix uninit memory read in undo code. Fix uint32_t in proto file.
Bram Moolenaar <bram@vim.org>
parents:
2214
diff
changeset
|
887 give_warning((char_u *)_("File contents changed, cannot use undo info"), TRUE); |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
888 verbose_leave(); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
889 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
890 goto error; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
891 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
892 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
893 /* Begin undo data for U */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
894 str_len = get4c(fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
895 if (str_len < 0) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
896 goto error; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
897 else if (str_len > 0) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
898 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
899 if ((line_ptr = U_ALLOC_LINE(str_len)) == NULL) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
900 goto error; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
901 for (i = 0; i < str_len; i++) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
902 line_ptr[i] = (char_u)getc(fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
903 line_ptr[i] = NUL; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
904 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
905 line_lnum = (linenr_T)get4c(fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
906 line_colnr = (colnr_T)get4c(fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
907 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
908 /* Begin general undo data */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
909 old_header_seq = get4c(fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
910 new_header_seq = get4c(fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
911 cur_header_seq = get4c(fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
912 num_head = get4c(fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
913 seq_last = get4c(fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
914 seq_cur = get4c(fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
915 seq_time = get4c(fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
916 |
2223
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
917 if (num_head < 0) |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
918 num_head = 0; |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
919 |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
920 /* uhp_table will store the freshly created undo headers we allocate |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
921 * until we insert them into curbuf. The table remains sorted by the |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
922 * sequence numbers of the headers. */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
923 uhp_table = (u_header_T **)U_ALLOC_LINE(num_head * sizeof(u_header_T *)); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
924 if (uhp_table == NULL) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
925 goto error; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
926 vim_memset(uhp_table, 0, num_head * sizeof(u_header_T *)); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
927 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
928 c = get2c(fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
929 while (c == UF_HEADER_MAGIC) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
930 { |
2223
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
931 if (num_read_uhps >= num_head) |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
932 { |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
933 EMSG2(_("E831 Undo file corruption: num_head: %s"), file_name); |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
934 u_free_uhp(uhp); |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
935 goto error; |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
936 } |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
937 |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
938 uhp = (u_header_T *)U_ALLOC_LINE((unsigned)sizeof(u_header_T)); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
939 if (uhp == NULL) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
940 goto error; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
941 vim_memset(uhp, 0, sizeof(u_header_T)); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
942 /* We're not actually trying to store pointers here. We're just storing |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
943 * IDs so we can swizzle them into pointers later - hence the type |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
944 * cast. */ |
2217
120502692d82
Improve the MS-Windows installer.
Bram Moolenaar <bram@vim.org>
parents:
2215
diff
changeset
|
945 uhp->uh_next = (u_header_T *)get4c(fp); |
120502692d82
Improve the MS-Windows installer.
Bram Moolenaar <bram@vim.org>
parents:
2215
diff
changeset
|
946 uhp->uh_prev = (u_header_T *)get4c(fp); |
120502692d82
Improve the MS-Windows installer.
Bram Moolenaar <bram@vim.org>
parents:
2215
diff
changeset
|
947 uhp->uh_alt_next = (u_header_T *)get4c(fp); |
120502692d82
Improve the MS-Windows installer.
Bram Moolenaar <bram@vim.org>
parents:
2215
diff
changeset
|
948 uhp->uh_alt_prev = (u_header_T *)get4c(fp); |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
949 uhp->uh_seq = get4c(fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
950 if (uhp->uh_seq <= 0) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
951 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
952 EMSG2(_("E825: Undo file corruption: invalid uh_seq.: %s"), |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
953 file_name); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
954 U_FREE_LINE(uhp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
955 goto error; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
956 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
957 uhp->uh_walk = 0; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
958 unserialize_pos(&uhp->uh_cursor, fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
959 #ifdef FEAT_VIRTUALEDIT |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
960 uhp->uh_cursor_vcol = get4c(fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
961 #else |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
962 (void)get4c(fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
963 #endif |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
964 uhp->uh_flags = get2c(fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
965 for (i = 0; i < NMARKS; ++i) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
966 unserialize_pos(&uhp->uh_namedm[i], fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
967 #ifdef FEAT_VISUAL |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
968 unserialize_visualinfo(&uhp->uh_visual, fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
969 #else |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
970 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
971 visualinfo_T info; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
972 unserialize_visualinfo(&info, fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
973 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
974 #endif |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
975 uhp->uh_time = get4c(fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
976 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
977 /* Unserialize uep list. The first 4 bytes is the length of the |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
978 * entire uep in bytes minus the length of the strings within. |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
979 * -1 is a sentinel value meaning no more ueps.*/ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
980 last_uep = NULL; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
981 while ((uep_len = get4c(fp)) != -1) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
982 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
983 uep = (u_entry_T *)U_ALLOC_LINE((unsigned)sizeof(u_entry_T)); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
984 if (uep == NULL) |
2223
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
985 { |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
986 u_free_uhp(uhp); |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
987 goto error; |
2223
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
988 } |
2215
cccb71c2c5c1
Fix uninit memory read in undo code. Fix uint32_t in proto file.
Bram Moolenaar <bram@vim.org>
parents:
2214
diff
changeset
|
989 vim_memset(uep, 0, sizeof(u_entry_T)); |
2223
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
990 if (last_uep == NULL) |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
991 uhp->uh_entry = uep; |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
992 else |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
993 last_uep->ue_next = uep; |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
994 last_uep = uep; |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
995 |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
996 uep->ue_top = get4c(fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
997 uep->ue_bot = get4c(fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
998 uep->ue_lcount = get4c(fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
999 uep->ue_size = get4c(fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1000 uep->ue_next = NULL; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1001 array = (char_u **)U_ALLOC_LINE( |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1002 (unsigned)(sizeof(char_u *) * uep->ue_size)); |
2223
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
1003 if (array == NULL) |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
1004 { |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
1005 u_free_uhp(uhp); |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
1006 goto error; |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
1007 } |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
1008 vim_memset(array, 0, sizeof(char_u *) * uep->ue_size); |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
1009 uep->ue_array = array; |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
1010 |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1011 for (i = 0; i < uep->ue_size; i++) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1012 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1013 line_len = get4c(fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1014 /* U_ALLOC_LINE provides an extra byte for the NUL terminator.*/ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1015 line = (char_u *)U_ALLOC_LINE( |
2223
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
1016 (unsigned)(sizeof(char_u) * line_len)); |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1017 if (line == NULL) |
2223
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
1018 { |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
1019 u_free_uhp(uhp); |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1020 goto error; |
2223
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
1021 } |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1022 for (j = 0; j < line_len; j++) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1023 line[j] = getc(fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1024 line[j] = '\0'; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1025 array[i] = line; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1026 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1027 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1028 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1029 /* Insertion sort the uhp into the table by its uh_seq. This is |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1030 * required because, while the number of uhps is limited to |
2223
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
1031 * num_head, and the uh_seq order is monotonic with respect to |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1032 * creation time, the starting uh_seq can be > 0 if any undolevel |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1033 * culling was done at undofile write time, and there can be uh_seq |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1034 * gaps in the uhps. |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1035 */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1036 for (i = num_read_uhps - 1; i >= -1; i--) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1037 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1038 /* if i == -1, we've hit the leftmost side of the table, so insert |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1039 * at uhp_table[0]. */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1040 if (i == -1 || uhp->uh_seq > uhp_table[i]->uh_seq) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1041 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1042 /* If we've had to move from the rightmost side of the table, |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1043 * we have to shift everything to the right by one spot. */ |
2219
c81f0a037059
And another small fix for persistent undo.
Bram Moolenaar <bram@vim.org>
parents:
2218
diff
changeset
|
1044 if (num_read_uhps - i - 1 > 0) |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1045 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1046 memmove(uhp_table + i + 2, uhp_table + i + 1, |
2223
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
1047 (num_read_uhps - i - 1) * sizeof(u_header_T *)); |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1048 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1049 uhp_table[i + 1] = uhp; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1050 break; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1051 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1052 else if (uhp->uh_seq == uhp_table[i]->uh_seq) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1053 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1054 EMSG2(_("E826 Undo file corruption: duplicate uh_seq: %s"), |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1055 file_name); |
2223
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
1056 u_free_uhp(uhp); |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1057 goto error; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1058 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1059 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1060 num_read_uhps++; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1061 c = get2c(fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1062 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1063 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1064 if (c != UF_END_MAGIC) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1065 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1066 EMSG2(_("E827: Undo file corruption; no end marker: %s"), file_name); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1067 goto error; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1068 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1069 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1070 /* We've organized all of the uhps into a table sorted by uh_seq. Now we |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1071 * iterate through the table and swizzle each sequence number we've |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1072 * stored in uh_foo into a pointer corresponding to the header with that |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1073 * sequence number. Then free curbuf's old undo structure, give curbuf |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1074 * the updated {old,new,cur}head pointers, and then free the table. */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1075 for (i = 0; i < num_head; i++) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1076 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1077 uhp = uhp_table[i]; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1078 if (uhp == NULL) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1079 continue; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1080 for (j = 0; j < num_head; j++) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1081 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1082 if (uhp_table[j] == NULL) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1083 continue; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1084 if (uhp_table[j]->uh_seq == (long)uhp->uh_next) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1085 uhp->uh_next = uhp_table[j]; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1086 if (uhp_table[j]->uh_seq == (long)uhp->uh_prev) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1087 uhp->uh_prev = uhp_table[j]; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1088 if (uhp_table[j]->uh_seq == (long)uhp->uh_alt_next) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1089 uhp->uh_alt_next = uhp_table[j]; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1090 if (uhp_table[j]->uh_seq == (long)uhp->uh_alt_prev) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1091 uhp->uh_alt_prev = uhp_table[j]; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1092 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1093 if (old_header_seq > 0 && old_idx < 0 && uhp->uh_seq == old_header_seq) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1094 old_idx = i; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1095 if (new_header_seq > 0 && new_idx < 0 && uhp->uh_seq == new_header_seq) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1096 new_idx = i; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1097 if (cur_header_seq > 0 && cur_idx < 0 && uhp->uh_seq == cur_header_seq) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1098 cur_idx = i; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1099 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1100 u_blockfree(curbuf); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1101 curbuf->b_u_oldhead = old_idx < 0 ? 0 : uhp_table[old_idx]; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1102 curbuf->b_u_newhead = new_idx < 0 ? 0 : uhp_table[new_idx]; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1103 curbuf->b_u_curhead = cur_idx < 0 ? 0 : uhp_table[cur_idx]; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1104 curbuf->b_u_line_ptr = line_ptr; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1105 curbuf->b_u_line_lnum = line_lnum; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1106 curbuf->b_u_line_colnr = line_colnr; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1107 curbuf->b_u_numhead = num_head; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1108 curbuf->b_u_seq_last = seq_last; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1109 curbuf->b_u_seq_cur = seq_cur; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1110 curbuf->b_u_seq_time = seq_time; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1111 U_FREE_LINE(uhp_table); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1112 #ifdef U_DEBUG |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1113 u_check(TRUE); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1114 #endif |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1115 if (name != NULL) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1116 smsg((char_u *)_("Finished reading undo file %s"), file_name); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1117 goto theend; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1118 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1119 error: |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1120 if (line_ptr != NULL) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1121 U_FREE_LINE(line_ptr); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1122 if (uhp_table != NULL) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1123 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1124 for (i = 0; i < num_head; i++) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1125 if (uhp_table[i] != NULL) |
2223
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
1126 u_free_uhp(uhp_table[i]); |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1127 U_FREE_LINE(uhp_table); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1128 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1129 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1130 theend: |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1131 if (fp != NULL) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1132 fclose(fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1133 if (file_name != name) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1134 vim_free(file_name); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1135 return; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1136 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1137 |
2223
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
1138 static void |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
1139 u_free_uhp(uhp) |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
1140 u_header_T *uhp; |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
1141 { |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
1142 u_entry_T *nuep; |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
1143 u_entry_T *uep; |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
1144 |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
1145 uep = uhp->uh_entry; |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
1146 while (uep != NULL) |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
1147 { |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
1148 nuep = uep->ue_next; |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
1149 u_freeentry(uep, uep->ue_size); |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
1150 uep = nuep; |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
1151 } |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
1152 U_FREE_LINE(uhp); |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
1153 } |
81b83a19e127
More strict checks for the undo file.
Bram Moolenaar <bram@vim.org>
parents:
2222
diff
changeset
|
1154 |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1155 /* |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1156 * Serialize "uep" to "fp". |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1157 */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1158 static int |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1159 serialize_uep(uep, fp) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1160 u_entry_T *uep; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1161 FILE *fp; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1162 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1163 int i; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1164 int uep_len; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1165 int *entry_lens; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1166 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1167 if (uep->ue_size > 0) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1168 entry_lens = (int *)alloc(uep->ue_size * sizeof(int)); |
2217
120502692d82
Improve the MS-Windows installer.
Bram Moolenaar <bram@vim.org>
parents:
2215
diff
changeset
|
1169 else |
120502692d82
Improve the MS-Windows installer.
Bram Moolenaar <bram@vim.org>
parents:
2215
diff
changeset
|
1170 entry_lens = NULL; |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1171 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1172 /* Define uep_len to be the size of the entire uep minus the size of its |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1173 * component strings, in bytes. The sizes of the component strings |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1174 * are written before each individual string. |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1175 * We have 4 entries each of 4 bytes, plus ue_size * 4 bytes |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1176 * of string size information. */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1177 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1178 uep_len = uep->ue_size * 4; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1179 /* Collect sizing information for later serialization. */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1180 for (i = 0; i < uep->ue_size; i++) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1181 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1182 entry_lens[i] = (int)STRLEN(uep->ue_array[i]); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1183 uep_len += entry_lens[i]; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1184 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1185 put_bytes(fp, (long_u)uep_len, 4); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1186 put_bytes(fp, (long_u)uep->ue_top, 4); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1187 put_bytes(fp, (long_u)uep->ue_bot, 4); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1188 put_bytes(fp, (long_u)uep->ue_lcount, 4); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1189 put_bytes(fp, (long_u)uep->ue_size, 4); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1190 for (i = 0; i < uep->ue_size; i++) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1191 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1192 if (put_bytes(fp, (long_u)entry_lens[i], 4) == FAIL) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1193 return FAIL; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1194 fprintf(fp, "%s", uep->ue_array[i]); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1195 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1196 if (uep->ue_size > 0) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1197 vim_free(entry_lens); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1198 return OK; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1199 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1200 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1201 /* |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1202 * Serialize "pos" to "fp". |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1203 */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1204 static void |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1205 serialize_pos(pos, fp) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1206 pos_T pos; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1207 FILE *fp; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1208 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1209 put_bytes(fp, (long_u)pos.lnum, 4); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1210 put_bytes(fp, (long_u)pos.col, 4); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1211 #ifdef FEAT_VIRTUALEDIT |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1212 put_bytes(fp, (long_u)pos.coladd, 4); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1213 #else |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1214 put_bytes(fp, (long_u)0, 4); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1215 #endif |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1216 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1217 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1218 /* |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1219 * Serialize "info" to "fp". |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1220 */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1221 static void |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1222 serialize_visualinfo(info, fp) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1223 visualinfo_T info; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1224 FILE *fp; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1225 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1226 serialize_pos(info.vi_start, fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1227 serialize_pos(info.vi_end, fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1228 put_bytes(fp, (long_u)info.vi_mode, 4); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1229 put_bytes(fp, (long_u)info.vi_curswant, 4); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1230 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1231 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1232 static char_u e_not_open[] = N_("E828: Cannot open undo file for writing: %s"); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1233 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1234 /* |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1235 * Write the undo tree in an undo file. |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1236 * When "name" is not NULL, use it as the name of the undo file. |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1237 * Otherwise use buf->b_ffname to generate the undo file name. |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1238 * "buf" must never be null, buf->b_ffname is used to obtain the original file |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1239 * permissions. |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1240 * "forceit" is TRUE for ":wundo!", FALSE otherwise. |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1241 * "hash[UNDO_HASH_SIZE]" must be the hash value of the buffer text. |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1242 */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1243 void |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1244 u_write_undo(name, forceit, buf, hash) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1245 char_u *name; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1246 int forceit; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1247 buf_T *buf; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1248 char_u *hash; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1249 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1250 u_header_T *uhp; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1251 u_entry_T *uep; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1252 char_u *file_name; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1253 int str_len, i, uep_len, mark; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1254 int fd; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1255 FILE *fp = NULL; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1256 int perm; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1257 int write_ok = FALSE; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1258 #ifdef UNIX |
2218
695ceebf17ca
Fix: :wundo didn't work in a buffer without a name.
Bram Moolenaar <bram@vim.org>
parents:
2217
diff
changeset
|
1259 int st_old_valid = FALSE; |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1260 struct stat st_old; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1261 struct stat st_new; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1262 #endif |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1263 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1264 if (name == NULL) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1265 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1266 file_name = u_get_undo_file_name(buf->b_ffname, FALSE); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1267 if (file_name == NULL) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1268 return; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1269 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1270 else |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1271 file_name = name; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1272 |
2218
695ceebf17ca
Fix: :wundo didn't work in a buffer without a name.
Bram Moolenaar <bram@vim.org>
parents:
2217
diff
changeset
|
1273 if (buf->b_ffname == NULL) |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1274 perm = 0600; |
2218
695ceebf17ca
Fix: :wundo didn't work in a buffer without a name.
Bram Moolenaar <bram@vim.org>
parents:
2217
diff
changeset
|
1275 else |
695ceebf17ca
Fix: :wundo didn't work in a buffer without a name.
Bram Moolenaar <bram@vim.org>
parents:
2217
diff
changeset
|
1276 { |
695ceebf17ca
Fix: :wundo didn't work in a buffer without a name.
Bram Moolenaar <bram@vim.org>
parents:
2217
diff
changeset
|
1277 #ifdef UNIX |
695ceebf17ca
Fix: :wundo didn't work in a buffer without a name.
Bram Moolenaar <bram@vim.org>
parents:
2217
diff
changeset
|
1278 if (mch_stat((char *)buf->b_ffname, &st_old) >= 0) |
695ceebf17ca
Fix: :wundo didn't work in a buffer without a name.
Bram Moolenaar <bram@vim.org>
parents:
2217
diff
changeset
|
1279 { |
695ceebf17ca
Fix: :wundo didn't work in a buffer without a name.
Bram Moolenaar <bram@vim.org>
parents:
2217
diff
changeset
|
1280 perm = st_old.st_mode; |
695ceebf17ca
Fix: :wundo didn't work in a buffer without a name.
Bram Moolenaar <bram@vim.org>
parents:
2217
diff
changeset
|
1281 st_old_valid = TRUE; |
695ceebf17ca
Fix: :wundo didn't work in a buffer without a name.
Bram Moolenaar <bram@vim.org>
parents:
2217
diff
changeset
|
1282 } |
695ceebf17ca
Fix: :wundo didn't work in a buffer without a name.
Bram Moolenaar <bram@vim.org>
parents:
2217
diff
changeset
|
1283 else |
695ceebf17ca
Fix: :wundo didn't work in a buffer without a name.
Bram Moolenaar <bram@vim.org>
parents:
2217
diff
changeset
|
1284 perm = 0600; |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1285 #else |
2218
695ceebf17ca
Fix: :wundo didn't work in a buffer without a name.
Bram Moolenaar <bram@vim.org>
parents:
2217
diff
changeset
|
1286 perm = mch_getperm(buf->b_ffname); |
695ceebf17ca
Fix: :wundo didn't work in a buffer without a name.
Bram Moolenaar <bram@vim.org>
parents:
2217
diff
changeset
|
1287 if (perm < 0) |
695ceebf17ca
Fix: :wundo didn't work in a buffer without a name.
Bram Moolenaar <bram@vim.org>
parents:
2217
diff
changeset
|
1288 perm = 0600; |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1289 #endif |
2218
695ceebf17ca
Fix: :wundo didn't work in a buffer without a name.
Bram Moolenaar <bram@vim.org>
parents:
2217
diff
changeset
|
1290 } |
695ceebf17ca
Fix: :wundo didn't work in a buffer without a name.
Bram Moolenaar <bram@vim.org>
parents:
2217
diff
changeset
|
1291 |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1292 /* set file protection same as original file, but strip s-bit */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1293 perm = perm & 0777; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1294 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1295 /* If the undo file exists, verify that it actually is an undo file, and |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1296 * delete it. */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1297 if (mch_getperm(file_name) >= 0) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1298 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1299 if (name == NULL || !forceit) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1300 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1301 /* Check we can read it and it's an undo file. */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1302 fd = mch_open((char *)file_name, O_RDONLY|O_EXTRA, 0); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1303 if (fd < 0) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1304 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1305 if (name != NULL || p_verbose > 0) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1306 smsg((char_u *)_("Will not overwrite with undo file, cannot read: %s"), |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1307 file_name); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1308 goto theend; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1309 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1310 else |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1311 { |
2222
271a5907f944
Fix for using ":wundo foo" when foo is an empty file.
Bram Moolenaar <bram@vim.org>
parents:
2219
diff
changeset
|
1312 char_u buf[2]; |
271a5907f944
Fix for using ":wundo foo" when foo is an empty file.
Bram Moolenaar <bram@vim.org>
parents:
2219
diff
changeset
|
1313 int len; |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1314 |
2222
271a5907f944
Fix for using ":wundo foo" when foo is an empty file.
Bram Moolenaar <bram@vim.org>
parents:
2219
diff
changeset
|
1315 len = vim_read(fd, buf, 2); |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1316 close(fd); |
2222
271a5907f944
Fix for using ":wundo foo" when foo is an empty file.
Bram Moolenaar <bram@vim.org>
parents:
2219
diff
changeset
|
1317 if (len < 2 || (buf[0] << 8) + buf[1] != UF_START_MAGIC) |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1318 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1319 if (name != NULL || p_verbose > 0) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1320 smsg((char_u *)_("Will not overwrite, this is not an undo file: %s"), |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1321 file_name); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1322 goto theend; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1323 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1324 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1325 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1326 mch_remove(file_name); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1327 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1328 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1329 fd = mch_open((char *)file_name, |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1330 O_CREAT|O_EXTRA|O_WRONLY|O_EXCL|O_NOFOLLOW, perm); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1331 (void)mch_setperm(file_name, perm); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1332 if (fd < 0) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1333 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1334 EMSG2(_(e_not_open), file_name); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1335 goto theend; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1336 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1337 if (p_verbose > 0) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1338 smsg((char_u *)_("Writing undo file: %s"), file_name); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1339 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1340 #ifdef UNIX |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1341 /* |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1342 * Try to set the group of the undo file same as the original file. If |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1343 * this fails, set the protection bits for the group same as the |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1344 * protection bits for others. |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1345 */ |
2218
695ceebf17ca
Fix: :wundo didn't work in a buffer without a name.
Bram Moolenaar <bram@vim.org>
parents:
2217
diff
changeset
|
1346 if (st_old_valid && (mch_stat((char *)file_name, &st_new) >= 0 |
695ceebf17ca
Fix: :wundo didn't work in a buffer without a name.
Bram Moolenaar <bram@vim.org>
parents:
2217
diff
changeset
|
1347 && st_new.st_gid != st_old.st_gid |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1348 # ifdef HAVE_FCHOWN /* sequent-ptx lacks fchown() */ |
2218
695ceebf17ca
Fix: :wundo didn't work in a buffer without a name.
Bram Moolenaar <bram@vim.org>
parents:
2217
diff
changeset
|
1349 && fchown(fd, (uid_t)-1, st_old.st_gid) != 0) |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1350 # endif |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1351 ) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1352 mch_setperm(file_name, (perm & 0707) | ((perm & 07) << 3)); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1353 # ifdef HAVE_SELINUX |
2218
695ceebf17ca
Fix: :wundo didn't work in a buffer without a name.
Bram Moolenaar <bram@vim.org>
parents:
2217
diff
changeset
|
1354 if (buf->b_ffname != NULL) |
695ceebf17ca
Fix: :wundo didn't work in a buffer without a name.
Bram Moolenaar <bram@vim.org>
parents:
2217
diff
changeset
|
1355 mch_copy_sec(buf->b_ffname, file_name); |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1356 # endif |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1357 #endif |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1358 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1359 fp = fdopen(fd, "w"); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1360 if (fp == NULL) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1361 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1362 EMSG2(_(e_not_open), file_name); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1363 close(fd); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1364 mch_remove(file_name); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1365 goto theend; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1366 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1367 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1368 /* Start writing, first overall file information */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1369 put_bytes(fp, (long_u)UF_START_MAGIC, 2); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1370 put_bytes(fp, (long_u)UF_VERSION, 2); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1371 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1372 /* Write a hash of the buffer text, so that we can verify it is still the |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1373 * same when reading the buffer text. */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1374 if (fwrite(hash, (size_t)UNDO_HASH_SIZE, (size_t)1, fp) != 1) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1375 goto write_error; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1376 put_bytes(fp, (long_u)buf->b_ml.ml_line_count, 4); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1377 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1378 /* Begin undo data for U */ |
2217
120502692d82
Improve the MS-Windows installer.
Bram Moolenaar <bram@vim.org>
parents:
2215
diff
changeset
|
1379 str_len = buf->b_u_line_ptr != NULL ? (int)STRLEN(buf->b_u_line_ptr) : 0; |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1380 put_bytes(fp, (long_u)str_len, 4); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1381 if (str_len > 0 && fwrite(buf->b_u_line_ptr, (size_t)str_len, |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1382 (size_t)1, fp) != 1) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1383 goto write_error; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1384 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1385 put_bytes(fp, (long_u)buf->b_u_line_lnum, 4); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1386 put_bytes(fp, (long_u)buf->b_u_line_colnr, 4); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1387 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1388 /* Begin general undo data */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1389 uhp = buf->b_u_oldhead; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1390 put_bytes(fp, (long_u)(uhp != NULL ? uhp->uh_seq : 0), 4); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1391 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1392 uhp = buf->b_u_newhead; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1393 put_bytes(fp, (long_u)(uhp != NULL ? uhp->uh_seq : 0), 4); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1394 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1395 uhp = buf->b_u_curhead; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1396 put_bytes(fp, (long_u)(uhp != NULL ? uhp->uh_seq : 0), 4); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1397 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1398 put_bytes(fp, (long_u)buf->b_u_numhead, 4); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1399 put_bytes(fp, (long_u)buf->b_u_seq_last, 4); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1400 put_bytes(fp, (long_u)buf->b_u_seq_cur, 4); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1401 put_bytes(fp, (long_u)buf->b_u_seq_time, 4); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1402 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1403 /* Iteratively serialize UHPs and their UEPs from the top down. */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1404 mark = ++lastmark; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1405 uhp = buf->b_u_oldhead; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1406 while (uhp != NULL) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1407 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1408 /* Serialize current UHP if we haven't seen it */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1409 if (uhp->uh_walk != mark) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1410 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1411 if (put_bytes(fp, (long_u)UF_HEADER_MAGIC, 2) == FAIL) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1412 goto write_error; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1413 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1414 put_bytes(fp, (long_u)((uhp->uh_next != NULL) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1415 ? uhp->uh_next->uh_seq : 0), 4); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1416 put_bytes(fp, (long_u)((uhp->uh_prev != NULL) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1417 ? uhp->uh_prev->uh_seq : 0), 4); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1418 put_bytes(fp, (long_u)((uhp->uh_alt_next != NULL) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1419 ? uhp->uh_alt_next->uh_seq : 0), 4); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1420 put_bytes(fp, (long_u)((uhp->uh_alt_prev != NULL) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1421 ? uhp->uh_alt_prev->uh_seq : 0), 4); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1422 put_bytes(fp, uhp->uh_seq, 4); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1423 serialize_pos(uhp->uh_cursor, fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1424 #ifdef FEAT_VIRTUALEDIT |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1425 put_bytes(fp, (long_u)uhp->uh_cursor_vcol, 4); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1426 #else |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1427 put_bytes(fp, (long_u)0, 4); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1428 #endif |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1429 put_bytes(fp, (long_u)uhp->uh_flags, 2); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1430 /* Assume NMARKS will stay the same. */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1431 for (i = 0; i < NMARKS; ++i) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1432 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1433 serialize_pos(uhp->uh_namedm[i], fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1434 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1435 #ifdef FEAT_VISUAL |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1436 serialize_visualinfo(uhp->uh_visual, fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1437 #endif |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1438 put_bytes(fp, (long_u)uhp->uh_time, 4); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1439 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1440 uep = uhp->uh_entry; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1441 while (uep != NULL) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1442 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1443 if (serialize_uep(uep, fp) == FAIL) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1444 goto write_error; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1445 uep = uep->ue_next; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1446 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1447 /* Sentinel value: no more ueps */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1448 uep_len = -1; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1449 put_bytes(fp, (long_u)uep_len, 4); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1450 uhp->uh_walk = mark; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1451 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1452 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1453 /* Now walk through the tree - algorithm from undo_time */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1454 if (uhp->uh_prev != NULL && uhp->uh_prev->uh_walk != mark) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1455 uhp = uhp->uh_prev; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1456 else if (uhp->uh_alt_next != NULL && uhp->uh_alt_next->uh_walk != mark) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1457 uhp = uhp->uh_alt_next; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1458 else if (uhp->uh_next != NULL && uhp->uh_alt_prev == NULL |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1459 && uhp->uh_next->uh_walk != mark) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1460 uhp = uhp->uh_next; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1461 else if (uhp->uh_alt_prev != NULL) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1462 uhp = uhp->uh_alt_prev; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1463 else |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1464 uhp = uhp->uh_next; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1465 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1466 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1467 if (put_bytes(fp, (long_u)UF_END_MAGIC, 2) == OK) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1468 write_ok = TRUE; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1469 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1470 write_error: |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1471 fclose(fp); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1472 if (!write_ok) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1473 EMSG2(_("E829: write error in undo file: %s"), file_name); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1474 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1475 #if defined(MACOS_CLASSIC) || defined(WIN3264) |
2218
695ceebf17ca
Fix: :wundo didn't work in a buffer without a name.
Bram Moolenaar <bram@vim.org>
parents:
2217
diff
changeset
|
1476 if (buf->b_ffname != NULL) |
695ceebf17ca
Fix: :wundo didn't work in a buffer without a name.
Bram Moolenaar <bram@vim.org>
parents:
2217
diff
changeset
|
1477 (void)mch_copy_file_attribute(buf->b_ffname, file_name); |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1478 #endif |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1479 #ifdef HAVE_ACL |
2218
695ceebf17ca
Fix: :wundo didn't work in a buffer without a name.
Bram Moolenaar <bram@vim.org>
parents:
2217
diff
changeset
|
1480 if (buf->b_ffname != NULL) |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1481 { |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1482 vim_acl_T acl; |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1483 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1484 /* For systems that support ACL: get the ACL from the original file. */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1485 acl = mch_get_acl(buf->b_ffname); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1486 mch_set_acl(file_name, acl); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1487 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1488 #endif |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1489 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1490 theend: |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1491 if (file_name != name) |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1492 vim_free(file_name); |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1493 } |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1494 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1495 #endif /* FEAT_PERSISTENT_UNDO */ |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1496 |
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1497 |
7 | 1498 /* |
1499 * If 'cpoptions' contains 'u': Undo the previous undo or redo (vi compatible). | |
1500 * If 'cpoptions' does not contain 'u': Always undo. | |
1501 */ | |
1502 void | |
1503 u_undo(count) | |
1504 int count; | |
1505 { | |
1506 /* | |
1507 * If we get an undo command while executing a macro, we behave like the | |
1508 * original vi. If this happens twice in one macro the result will not | |
1509 * be compatible. | |
1510 */ | |
1511 if (curbuf->b_u_synced == FALSE) | |
1512 { | |
825 | 1513 u_sync(TRUE); |
7 | 1514 count = 1; |
1515 } | |
1516 | |
1517 if (vim_strchr(p_cpo, CPO_UNDO) == NULL) | |
1518 undo_undoes = TRUE; | |
1519 else | |
1520 undo_undoes = !undo_undoes; | |
1521 u_doit(count); | |
1522 } | |
1523 | |
1524 /* | |
1525 * If 'cpoptions' contains 'u': Repeat the previous undo or redo. | |
1526 * If 'cpoptions' does not contain 'u': Always redo. | |
1527 */ | |
1528 void | |
1529 u_redo(count) | |
1530 int count; | |
1531 { | |
1532 if (vim_strchr(p_cpo, CPO_UNDO) == NULL) | |
1533 undo_undoes = FALSE; | |
1534 u_doit(count); | |
1535 } | |
1536 | |
1537 /* | |
1538 * Undo or redo, depending on 'undo_undoes', 'count' times. | |
1539 */ | |
1540 static void | |
777 | 1541 u_doit(startcount) |
1542 int startcount; | |
7 | 1543 { |
777 | 1544 int count = startcount; |
1545 | |
632 | 1546 if (!undo_allowed()) |
7 | 1547 return; |
1548 | |
1549 u_newcount = 0; | |
1550 u_oldcount = 0; | |
179 | 1551 if (curbuf->b_ml.ml_flags & ML_EMPTY) |
1552 u_oldcount = -1; | |
7 | 1553 while (count--) |
1554 { | |
1555 if (undo_undoes) | |
1556 { | |
1557 if (curbuf->b_u_curhead == NULL) /* first undo */ | |
1558 curbuf->b_u_curhead = curbuf->b_u_newhead; | |
1559 else if (p_ul > 0) /* multi level undo */ | |
1560 /* get next undo */ | |
1561 curbuf->b_u_curhead = curbuf->b_u_curhead->uh_next; | |
1562 /* nothing to undo */ | |
1563 if (curbuf->b_u_numhead == 0 || curbuf->b_u_curhead == NULL) | |
1564 { | |
1565 /* stick curbuf->b_u_curhead at end */ | |
1566 curbuf->b_u_curhead = curbuf->b_u_oldhead; | |
1567 beep_flush(); | |
777 | 1568 if (count == startcount - 1) |
1569 { | |
1570 MSG(_("Already at oldest change")); | |
1571 return; | |
1572 } | |
7 | 1573 break; |
1574 } | |
1575 | |
777 | 1576 u_undoredo(TRUE); |
7 | 1577 } |
1578 else | |
1579 { | |
1580 if (curbuf->b_u_curhead == NULL || p_ul <= 0) | |
1581 { | |
1582 beep_flush(); /* nothing to redo */ | |
777 | 1583 if (count == startcount - 1) |
1584 { | |
1585 MSG(_("Already at newest change")); | |
1586 return; | |
1587 } | |
7 | 1588 break; |
1589 } | |
1590 | |
777 | 1591 u_undoredo(FALSE); |
753 | 1592 |
1593 /* Advance for next redo. Set "newhead" when at the end of the | |
1594 * redoable changes. */ | |
1595 if (curbuf->b_u_curhead->uh_prev == NULL) | |
1596 curbuf->b_u_newhead = curbuf->b_u_curhead; | |
7 | 1597 curbuf->b_u_curhead = curbuf->b_u_curhead->uh_prev; |
1598 } | |
1599 } | |
798 | 1600 u_undo_end(undo_undoes, FALSE); |
753 | 1601 } |
1602 | |
1603 /* | |
1604 * Undo or redo over the timeline. | |
1605 * When "step" is negative go back in time, otherwise goes forward in time. | |
758 | 1606 * When "sec" is FALSE make "step" steps, when "sec" is TRUE use "step" as |
1607 * seconds. | |
772 | 1608 * When "absolute" is TRUE use "step" as the sequence number to jump to. |
1609 * "sec" must be FALSE then. | |
753 | 1610 */ |
1611 void | |
772 | 1612 undo_time(step, sec, absolute) |
758 | 1613 long step; |
1614 int sec; | |
772 | 1615 int absolute; |
753 | 1616 { |
1617 long target; | |
1618 long closest; | |
758 | 1619 long closest_start; |
1620 long closest_seq = 0; | |
1621 long val; | |
753 | 1622 u_header_T *uhp; |
1623 u_header_T *last; | |
1624 int mark; | |
1625 int nomark; | |
1626 int round; | |
758 | 1627 int dosec = sec; |
1628 int above = FALSE; | |
794 | 1629 int did_undo = TRUE; |
753 | 1630 |
766 | 1631 /* First make sure the current undoable change is synced. */ |
1632 if (curbuf->b_u_synced == FALSE) | |
825 | 1633 u_sync(TRUE); |
766 | 1634 |
753 | 1635 u_newcount = 0; |
1636 u_oldcount = 0; | |
179 | 1637 if (curbuf->b_ml.ml_flags & ML_EMPTY) |
753 | 1638 u_oldcount = -1; |
1639 | |
777 | 1640 /* "target" is the node below which we want to be. |
1641 * Init "closest" to a value we can't reach. */ | |
772 | 1642 if (absolute) |
1643 { | |
1644 target = step; | |
777 | 1645 closest = -1; |
753 | 1646 } |
1647 else | |
1648 { | |
777 | 1649 /* When doing computations with time_t subtract starttime, because |
1650 * time_t converted to a long may result in a wrong number. */ | |
758 | 1651 if (sec) |
777 | 1652 target = (long)(curbuf->b_u_seq_time - starttime) + step; |
1653 else | |
1654 target = curbuf->b_u_seq_cur + step; | |
1655 if (step < 0) | |
758 | 1656 { |
777 | 1657 if (target < 0) |
1658 target = 0; | |
1659 closest = -1; | |
758 | 1660 } |
1661 else | |
1662 { | |
777 | 1663 if (sec) |
798 | 1664 closest = (long)(time(NULL) - starttime + 1); |
777 | 1665 else |
1666 closest = curbuf->b_u_seq_last + 2; | |
1667 if (target >= closest) | |
1668 target = closest - 1; | |
758 | 1669 } |
753 | 1670 } |
758 | 1671 closest_start = closest; |
777 | 1672 closest_seq = curbuf->b_u_seq_cur; |
753 | 1673 |
758 | 1674 /* |
1675 * May do this twice: | |
753 | 1676 * 1. Search for "target", update "closest" to the best match found. |
758 | 1677 * 2. If "target" not found search for "closest". |
1678 * | |
1679 * When using the closest time we use the sequence number in the second | |
1680 * round, because there may be several entries with the same time. | |
1681 */ | |
753 | 1682 for (round = 1; round <= 2; ++round) |
1683 { | |
1684 /* Find the path from the current state to where we want to go. The | |
1685 * desired state can be anywhere in the undo tree, need to go all over | |
1686 * it. We put "nomark" in uh_walk where we have been without success, | |
1687 * "mark" where it could possibly be. */ | |
1688 mark = ++lastmark; | |
1689 nomark = ++lastmark; | |
1690 | |
1691 if (curbuf->b_u_curhead == NULL) /* at leaf of the tree */ | |
1692 uhp = curbuf->b_u_newhead; | |
1693 else | |
1694 uhp = curbuf->b_u_curhead; | |
1695 | |
1696 while (uhp != NULL) | |
1697 { | |
1698 uhp->uh_walk = mark; | |
798 | 1699 val = (long)(dosec ? (uhp->uh_time - starttime) : uhp->uh_seq); |
753 | 1700 |
758 | 1701 if (round == 1) |
1702 { | |
1703 /* Remember the header that is closest to the target. | |
1704 * It must be at least in the right direction (checked with | |
777 | 1705 * "b_u_seq_cur"). When the timestamp is equal find the |
758 | 1706 * highest/lowest sequence number. */ |
777 | 1707 if ((step < 0 ? uhp->uh_seq <= curbuf->b_u_seq_cur |
1708 : uhp->uh_seq > curbuf->b_u_seq_cur) | |
1709 && ((dosec && val == closest) | |
1710 ? (step < 0 | |
1711 ? uhp->uh_seq < closest_seq | |
1712 : uhp->uh_seq > closest_seq) | |
1713 : closest == closest_start | |
1714 || (val > target | |
1715 ? (closest > target | |
1716 ? val - target <= closest - target | |
1717 : val - target <= target - closest) | |
1718 : (closest > target | |
1719 ? target - val <= closest - target | |
1720 : target - val <= target - closest)))) | |
758 | 1721 { |
1722 closest = val; | |
1723 closest_seq = uhp->uh_seq; | |
1724 } | |
1725 } | |
1726 | |
1727 /* Quit searching when we found a match. But when searching for a | |
1728 * time we need to continue looking for the best uh_seq. */ | |
1729 if (target == val && !dosec) | |
1730 break; | |
753 | 1731 |
1732 /* go down in the tree if we haven't been there */ | |
1733 if (uhp->uh_prev != NULL && uhp->uh_prev->uh_walk != nomark | |
1734 && uhp->uh_prev->uh_walk != mark) | |
1735 uhp = uhp->uh_prev; | |
1736 | |
1737 /* go to alternate branch if we haven't been there */ | |
1738 else if (uhp->uh_alt_next != NULL | |
1739 && uhp->uh_alt_next->uh_walk != nomark | |
1740 && uhp->uh_alt_next->uh_walk != mark) | |
1741 uhp = uhp->uh_alt_next; | |
1742 | |
1743 /* go up in the tree if we haven't been there and we are at the | |
1744 * start of alternate branches */ | |
1745 else if (uhp->uh_next != NULL && uhp->uh_alt_prev == NULL | |
1746 && uhp->uh_next->uh_walk != nomark | |
1747 && uhp->uh_next->uh_walk != mark) | |
798 | 1748 { |
1749 /* If still at the start we don't go through this change. */ | |
1750 if (uhp == curbuf->b_u_curhead) | |
1751 uhp->uh_walk = nomark; | |
753 | 1752 uhp = uhp->uh_next; |
798 | 1753 } |
753 | 1754 |
1755 else | |
1756 { | |
1757 /* need to backtrack; mark this node as useless */ | |
1758 uhp->uh_walk = nomark; | |
1759 if (uhp->uh_alt_prev != NULL) | |
1760 uhp = uhp->uh_alt_prev; | |
1761 else | |
1762 uhp = uhp->uh_next; | |
1763 } | |
1764 } | |
1765 | |
1766 if (uhp != NULL) /* found it */ | |
1767 break; | |
772 | 1768 |
1769 if (absolute) | |
1770 { | |
2214
f8222d1f9a73
Included patch for persistent undo. Lots of changes and added test.
Bram Moolenaar <bram@vim.org>
parents:
2210
diff
changeset
|
1771 EMSGN(_("E830: Undo number %ld not found"), step); |
772 | 1772 return; |
1773 } | |
1774 | |
758 | 1775 if (closest == closest_start) |
753 | 1776 { |
758 | 1777 if (step < 0) |
1778 MSG(_("Already at oldest change")); | |
1779 else | |
1780 MSG(_("Already at newest change")); | |
753 | 1781 return; |
1782 } | |
1783 | |
758 | 1784 target = closest_seq; |
1785 dosec = FALSE; | |
1786 if (step < 0) | |
1787 above = TRUE; /* stop above the header */ | |
753 | 1788 } |
1789 | |
1790 /* If we found it: Follow the path to go to where we want to be. */ | |
1791 if (uhp != NULL) | |
1792 { | |
1793 /* | |
1794 * First go up the tree as much as needed. | |
1795 */ | |
1796 for (;;) | |
1797 { | |
1798 uhp = curbuf->b_u_curhead; | |
1799 if (uhp == NULL) | |
1800 uhp = curbuf->b_u_newhead; | |
1801 else | |
1802 uhp = uhp->uh_next; | |
777 | 1803 if (uhp == NULL || uhp->uh_walk != mark |
1804 || (uhp->uh_seq == target && !above)) | |
753 | 1805 break; |
1806 curbuf->b_u_curhead = uhp; | |
777 | 1807 u_undoredo(TRUE); |
753 | 1808 uhp->uh_walk = nomark; /* don't go back down here */ |
1809 } | |
1810 | |
1811 /* | |
758 | 1812 * And now go down the tree (redo), branching off where needed. |
753 | 1813 */ |
1814 uhp = curbuf->b_u_curhead; | |
777 | 1815 while (uhp != NULL) |
753 | 1816 { |
1056 | 1817 /* Go back to the first branch with a mark. */ |
1818 while (uhp->uh_alt_prev != NULL | |
1819 && uhp->uh_alt_prev->uh_walk == mark) | |
1820 uhp = uhp->uh_alt_prev; | |
1821 | |
753 | 1822 /* Find the last branch with a mark, that's the one. */ |
1823 last = uhp; | |
1824 while (last->uh_alt_next != NULL | |
1825 && last->uh_alt_next->uh_walk == mark) | |
1826 last = last->uh_alt_next; | |
1827 if (last != uhp) | |
1828 { | |
1829 /* Make the used branch the first entry in the list of | |
1830 * alternatives to make "u" and CTRL-R take this branch. */ | |
1056 | 1831 while (uhp->uh_alt_prev != NULL) |
1832 uhp = uhp->uh_alt_prev; | |
753 | 1833 if (last->uh_alt_next != NULL) |
1834 last->uh_alt_next->uh_alt_prev = last->uh_alt_prev; | |
1835 last->uh_alt_prev->uh_alt_next = last->uh_alt_next; | |
1836 last->uh_alt_prev = NULL; | |
1837 last->uh_alt_next = uhp; | |
1838 uhp->uh_alt_prev = last; | |
1839 | |
1840 uhp = last; | |
777 | 1841 if (uhp->uh_next != NULL) |
1842 uhp->uh_next->uh_prev = uhp; | |
753 | 1843 } |
758 | 1844 curbuf->b_u_curhead = uhp; |
753 | 1845 |
1846 if (uhp->uh_walk != mark) | |
1847 break; /* must have reached the target */ | |
1848 | |
758 | 1849 /* Stop when going backwards in time and didn't find the exact |
1850 * header we were looking for. */ | |
1851 if (uhp->uh_seq == target && above) | |
798 | 1852 { |
1853 curbuf->b_u_seq_cur = target - 1; | |
758 | 1854 break; |
798 | 1855 } |
758 | 1856 |
777 | 1857 u_undoredo(FALSE); |
753 | 1858 |
1859 /* Advance "curhead" to below the header we last used. If it | |
1860 * becomes NULL then we need to set "newhead" to this leaf. */ | |
1861 if (uhp->uh_prev == NULL) | |
1862 curbuf->b_u_newhead = uhp; | |
1863 curbuf->b_u_curhead = uhp->uh_prev; | |
794 | 1864 did_undo = FALSE; |
753 | 1865 |
1866 if (uhp->uh_seq == target) /* found it! */ | |
1867 break; | |
1868 | |
1869 uhp = uhp->uh_prev; | |
1870 if (uhp == NULL || uhp->uh_walk != mark) | |
1871 { | |
777 | 1872 /* Need to redo more but can't find it... */ |
753 | 1873 EMSG2(_(e_intern2), "undo_time()"); |
1874 break; | |
1875 } | |
1876 } | |
1877 } | |
798 | 1878 u_undo_end(did_undo, absolute); |
7 | 1879 } |
1880 | |
1881 /* | |
1882 * u_undoredo: common code for undo and redo | |
1883 * | |
1884 * The lines in the file are replaced by the lines in the entry list at | |
1885 * curbuf->b_u_curhead. The replaced lines in the file are saved in the entry | |
1886 * list for the next undo/redo. | |
777 | 1887 * |
1888 * When "undo" is TRUE we go up in the tree, when FALSE we go down. | |
7 | 1889 */ |
1890 static void | |
777 | 1891 u_undoredo(undo) |
1892 int undo; | |
7 | 1893 { |
1894 char_u **newarray = NULL; | |
1895 linenr_T oldsize; | |
1896 linenr_T newsize; | |
1897 linenr_T top, bot; | |
1898 linenr_T lnum; | |
1899 linenr_T newlnum = MAXLNUM; | |
1900 long i; | |
1901 u_entry_T *uep, *nuep; | |
1902 u_entry_T *newlist = NULL; | |
1903 int old_flags; | |
1904 int new_flags; | |
1905 pos_T namedm[NMARKS]; | |
692 | 1906 #ifdef FEAT_VISUAL |
1907 visualinfo_T visualinfo; | |
1908 #endif | |
7 | 1909 int empty_buffer; /* buffer became empty */ |
758 | 1910 u_header_T *curhead = curbuf->b_u_curhead; |
7 | 1911 |
1415 | 1912 #ifdef U_DEBUG |
1913 u_check(FALSE); | |
1914 #endif | |
758 | 1915 old_flags = curhead->uh_flags; |
7 | 1916 new_flags = (curbuf->b_changed ? UH_CHANGED : 0) + |
1917 ((curbuf->b_ml.ml_flags & ML_EMPTY) ? UH_EMPTYBUF : 0); | |
1918 setpcmark(); | |
1919 | |
1920 /* | |
1921 * save marks before undo/redo | |
1922 */ | |
1923 mch_memmove(namedm, curbuf->b_namedm, sizeof(pos_T) * NMARKS); | |
692 | 1924 #ifdef FEAT_VISUAL |
1925 visualinfo = curbuf->b_visual; | |
1926 #endif | |
7 | 1927 curbuf->b_op_start.lnum = curbuf->b_ml.ml_line_count; |
1928 curbuf->b_op_start.col = 0; | |
1929 curbuf->b_op_end.lnum = 0; | |
1930 curbuf->b_op_end.col = 0; | |
1931 | |
758 | 1932 for (uep = curhead->uh_entry; uep != NULL; uep = nuep) |
7 | 1933 { |
1934 top = uep->ue_top; | |
1935 bot = uep->ue_bot; | |
1936 if (bot == 0) | |
1937 bot = curbuf->b_ml.ml_line_count + 1; | |
1938 if (top > curbuf->b_ml.ml_line_count || top >= bot | |
1939 || bot > curbuf->b_ml.ml_line_count + 1) | |
1940 { | |
1941 EMSG(_("E438: u_undo: line numbers wrong")); | |
1942 changed(); /* don't want UNCHANGED now */ | |
1943 return; | |
1944 } | |
1945 | |
1946 oldsize = bot - top - 1; /* number of lines before undo */ | |
1947 newsize = uep->ue_size; /* number of lines after undo */ | |
1948 | |
1949 if (top < newlnum) | |
1950 { | |
1951 /* If the saved cursor is somewhere in this undo block, move it to | |
1952 * the remembered position. Makes "gwap" put the cursor back | |
1953 * where it was. */ | |
758 | 1954 lnum = curhead->uh_cursor.lnum; |
7 | 1955 if (lnum >= top && lnum <= top + newsize + 1) |
1956 { | |
758 | 1957 curwin->w_cursor = curhead->uh_cursor; |
7 | 1958 newlnum = curwin->w_cursor.lnum - 1; |
1959 } | |
1960 else | |
1961 { | |
1962 /* Use the first line that actually changed. Avoids that | |
1963 * undoing auto-formatting puts the cursor in the previous | |
1964 * line. */ | |
1965 for (i = 0; i < newsize && i < oldsize; ++i) | |
1966 if (STRCMP(uep->ue_array[i], ml_get(top + 1 + i)) != 0) | |
1967 break; | |
1968 if (i == newsize && newlnum == MAXLNUM && uep->ue_next == NULL) | |
1969 { | |
1970 newlnum = top; | |
1971 curwin->w_cursor.lnum = newlnum + 1; | |
1972 } | |
1973 else if (i < newsize) | |
1974 { | |
1975 newlnum = top + i; | |
1976 curwin->w_cursor.lnum = newlnum + 1; | |
1977 } | |
1978 } | |
1979 } | |
1980 | |
1981 empty_buffer = FALSE; | |
1982 | |
1983 /* delete the lines between top and bot and save them in newarray */ | |
168 | 1984 if (oldsize > 0) |
7 | 1985 { |
168 | 1986 if ((newarray = (char_u **)U_ALLOC_LINE( |
7 | 1987 (unsigned)(sizeof(char_u *) * oldsize))) == NULL) |
1988 { | |
1989 do_outofmem_msg((long_u)(sizeof(char_u *) * oldsize)); | |
1990 /* | |
1991 * We have messed up the entry list, repair is impossible. | |
1992 * we have to free the rest of the list. | |
1993 */ | |
1994 while (uep != NULL) | |
1995 { | |
1996 nuep = uep->ue_next; | |
1997 u_freeentry(uep, uep->ue_size); | |
1998 uep = nuep; | |
1999 } | |
2000 break; | |
2001 } | |
2002 /* delete backwards, it goes faster in most cases */ | |
2003 for (lnum = bot - 1, i = oldsize; --i >= 0; --lnum) | |
2004 { | |
2005 /* what can we do when we run out of memory? */ | |
2006 if ((newarray[i] = u_save_line(lnum)) == NULL) | |
2007 do_outofmem_msg((long_u)0); | |
2008 /* remember we deleted the last line in the buffer, and a | |
2009 * dummy empty line will be inserted */ | |
2010 if (curbuf->b_ml.ml_line_count == 1) | |
2011 empty_buffer = TRUE; | |
2012 ml_delete(lnum, FALSE); | |
2013 } | |
2014 } | |
414 | 2015 else |
2016 newarray = NULL; | |
7 | 2017 |
2018 /* insert the lines in u_array between top and bot */ | |
2019 if (newsize) | |
2020 { | |
2021 for (lnum = top, i = 0; i < newsize; ++i, ++lnum) | |
2022 { | |
2023 /* | |
2024 * If the file is empty, there is an empty line 1 that we | |
2025 * should get rid of, by replacing it with the new line | |
2026 */ | |
2027 if (empty_buffer && lnum == 0) | |
2028 ml_replace((linenr_T)1, uep->ue_array[i], TRUE); | |
2029 else | |
2030 ml_append(lnum, uep->ue_array[i], (colnr_T)0, FALSE); | |
168 | 2031 U_FREE_LINE(uep->ue_array[i]); |
7 | 2032 } |
168 | 2033 U_FREE_LINE((char_u *)uep->ue_array); |
7 | 2034 } |
2035 | |
2036 /* adjust marks */ | |
2037 if (oldsize != newsize) | |
2038 { | |
2039 mark_adjust(top + 1, top + oldsize, (long)MAXLNUM, | |
2040 (long)newsize - (long)oldsize); | |
2041 if (curbuf->b_op_start.lnum > top + oldsize) | |
2042 curbuf->b_op_start.lnum += newsize - oldsize; | |
2043 if (curbuf->b_op_end.lnum > top + oldsize) | |
2044 curbuf->b_op_end.lnum += newsize - oldsize; | |
2045 } | |
2046 | |
2047 changed_lines(top + 1, 0, bot, newsize - oldsize); | |
2048 | |
2049 /* set '[ and '] mark */ | |
2050 if (top + 1 < curbuf->b_op_start.lnum) | |
2051 curbuf->b_op_start.lnum = top + 1; | |
2052 if (newsize == 0 && top + 1 > curbuf->b_op_end.lnum) | |
2053 curbuf->b_op_end.lnum = top + 1; | |
2054 else if (top + newsize > curbuf->b_op_end.lnum) | |
2055 curbuf->b_op_end.lnum = top + newsize; | |
2056 | |
2057 u_newcount += newsize; | |
2058 u_oldcount += oldsize; | |
2059 uep->ue_size = oldsize; | |
2060 uep->ue_array = newarray; | |
2061 uep->ue_bot = top + newsize + 1; | |
2062 | |
2063 /* | |
2064 * insert this entry in front of the new entry list | |
2065 */ | |
2066 nuep = uep->ue_next; | |
2067 uep->ue_next = newlist; | |
2068 newlist = uep; | |
2069 } | |
2070 | |
758 | 2071 curhead->uh_entry = newlist; |
2072 curhead->uh_flags = new_flags; | |
7 | 2073 if ((old_flags & UH_EMPTYBUF) && bufempty()) |
2074 curbuf->b_ml.ml_flags |= ML_EMPTY; | |
2075 if (old_flags & UH_CHANGED) | |
2076 changed(); | |
2077 else | |
33 | 2078 #ifdef FEAT_NETBEANS_INTG |
2079 /* per netbeans undo rules, keep it as modified */ | |
2080 if (!isNetbeansModified(curbuf)) | |
2081 #endif | |
7 | 2082 unchanged(curbuf, FALSE); |
2083 | |
2084 /* | |
2085 * restore marks from before undo/redo | |
2086 */ | |
2087 for (i = 0; i < NMARKS; ++i) | |
758 | 2088 if (curhead->uh_namedm[i].lnum != 0) |
7 | 2089 { |
758 | 2090 curbuf->b_namedm[i] = curhead->uh_namedm[i]; |
2091 curhead->uh_namedm[i] = namedm[i]; | |
7 | 2092 } |
692 | 2093 #ifdef FEAT_VISUAL |
758 | 2094 if (curhead->uh_visual.vi_start.lnum != 0) |
692 | 2095 { |
758 | 2096 curbuf->b_visual = curhead->uh_visual; |
2097 curhead->uh_visual = visualinfo; | |
692 | 2098 } |
2099 #endif | |
7 | 2100 |
2101 /* | |
2102 * If the cursor is only off by one line, put it at the same position as | |
2103 * before starting the change (for the "o" command). | |
2104 * Otherwise the cursor should go to the first undone line. | |
2105 */ | |
758 | 2106 if (curhead->uh_cursor.lnum + 1 == curwin->w_cursor.lnum |
7 | 2107 && curwin->w_cursor.lnum > 1) |
2108 --curwin->w_cursor.lnum; | |
758 | 2109 if (curhead->uh_cursor.lnum == curwin->w_cursor.lnum) |
7 | 2110 { |
758 | 2111 curwin->w_cursor.col = curhead->uh_cursor.col; |
7 | 2112 #ifdef FEAT_VIRTUALEDIT |
758 | 2113 if (virtual_active() && curhead->uh_cursor_vcol >= 0) |
2114 coladvance((colnr_T)curhead->uh_cursor_vcol); | |
7 | 2115 else |
2116 curwin->w_cursor.coladd = 0; | |
2117 #endif | |
2118 } | |
2119 else if (curwin->w_cursor.lnum <= curbuf->b_ml.ml_line_count) | |
2120 beginline(BL_SOL | BL_FIX); | |
2121 else | |
2122 { | |
2123 /* We get here with the current cursor line being past the end (eg | |
2124 * after adding lines at the end of the file, and then undoing it). | |
2125 * check_cursor() will move the cursor to the last line. Move it to | |
2126 * the first column here. */ | |
2127 curwin->w_cursor.col = 0; | |
2128 #ifdef FEAT_VIRTUALEDIT | |
2129 curwin->w_cursor.coladd = 0; | |
2130 #endif | |
2131 } | |
2132 | |
2133 /* Make sure the cursor is on an existing line and column. */ | |
2134 check_cursor(); | |
758 | 2135 |
2136 /* Remember where we are for "g-" and ":earlier 10s". */ | |
2137 curbuf->b_u_seq_cur = curhead->uh_seq; | |
777 | 2138 if (undo) |
2139 /* We are below the previous undo. However, to make ":earlier 1s" | |
2140 * work we compute this as being just above the just undone change. */ | |
2141 --curbuf->b_u_seq_cur; | |
2142 | |
2143 /* The timestamp can be the same for multiple changes, just use the one of | |
2144 * the undone/redone change. */ | |
758 | 2145 curbuf->b_u_seq_time = curhead->uh_time; |
1415 | 2146 #ifdef U_DEBUG |
2147 u_check(FALSE); | |
2148 #endif | |
7 | 2149 } |
2150 | |
2151 /* | |
2152 * If we deleted or added lines, report the number of less/more lines. | |
2153 * Otherwise, report the number of changes (this may be incorrect | |
2154 * in some cases, but it's better than nothing). | |
2155 */ | |
2156 static void | |
798 | 2157 u_undo_end(did_undo, absolute) |
794 | 2158 int did_undo; /* just did an undo */ |
798 | 2159 int absolute; /* used ":undo N" */ |
7 | 2160 { |
944 | 2161 char *msgstr; |
772 | 2162 u_header_T *uhp; |
2163 char_u msgbuf[80]; | |
753 | 2164 |
7 | 2165 #ifdef FEAT_FOLDING |
2166 if ((fdo_flags & FDO_UNDO) && KeyTyped) | |
2167 foldOpenCursor(); | |
2168 #endif | |
753 | 2169 |
2170 if (global_busy /* no messages now, wait until global is finished */ | |
2171 || !messaging()) /* 'lazyredraw' set, don't do messages now */ | |
2172 return; | |
2173 | |
2174 if (curbuf->b_ml.ml_flags & ML_EMPTY) | |
2175 --u_newcount; | |
2176 | |
2177 u_oldcount -= u_newcount; | |
2178 if (u_oldcount == -1) | |
944 | 2179 msgstr = N_("more line"); |
753 | 2180 else if (u_oldcount < 0) |
944 | 2181 msgstr = N_("more lines"); |
753 | 2182 else if (u_oldcount == 1) |
944 | 2183 msgstr = N_("line less"); |
753 | 2184 else if (u_oldcount > 1) |
944 | 2185 msgstr = N_("fewer lines"); |
753 | 2186 else |
2187 { | |
2188 u_oldcount = u_newcount; | |
2189 if (u_newcount == 1) | |
944 | 2190 msgstr = N_("change"); |
753 | 2191 else |
944 | 2192 msgstr = N_("changes"); |
753 | 2193 } |
2194 | |
772 | 2195 if (curbuf->b_u_curhead != NULL) |
794 | 2196 { |
798 | 2197 /* For ":undo N" we prefer a "after #N" message. */ |
2198 if (absolute && curbuf->b_u_curhead->uh_next != NULL) | |
2199 { | |
2200 uhp = curbuf->b_u_curhead->uh_next; | |
2201 did_undo = FALSE; | |
2202 } | |
2203 else if (did_undo) | |
794 | 2204 uhp = curbuf->b_u_curhead; |
2205 else | |
2206 uhp = curbuf->b_u_curhead->uh_next; | |
2207 } | |
753 | 2208 else |
772 | 2209 uhp = curbuf->b_u_newhead; |
753 | 2210 |
772 | 2211 if (uhp == NULL) |
2212 *msgbuf = NUL; | |
2213 else | |
2214 u_add_time(msgbuf, sizeof(msgbuf), uhp->uh_time); | |
2215 | |
794 | 2216 smsg((char_u *)_("%ld %s; %s #%ld %s"), |
777 | 2217 u_oldcount < 0 ? -u_oldcount : u_oldcount, |
944 | 2218 _(msgstr), |
794 | 2219 did_undo ? _("before") : _("after"), |
2220 uhp == NULL ? 0L : uhp->uh_seq, | |
2221 msgbuf); | |
7 | 2222 } |
2223 | |
2224 /* | |
2225 * u_sync: stop adding to the current entry list | |
2226 */ | |
2227 void | |
825 | 2228 u_sync(force) |
2229 int force; /* Also sync when no_u_sync is set. */ | |
7 | 2230 { |
825 | 2231 /* Skip it when already synced or syncing is disabled. */ |
2232 if (curbuf->b_u_synced || (!force && no_u_sync > 0)) | |
2233 return; | |
7 | 2234 #if defined(FEAT_XIM) && defined(FEAT_GUI_GTK) |
2235 if (im_is_preediting()) | |
2236 return; /* XIM is busy, don't break an undo sequence */ | |
2237 #endif | |
2238 if (p_ul < 0) | |
2239 curbuf->b_u_synced = TRUE; /* no entries, nothing to do */ | |
2240 else | |
2241 { | |
2242 u_getbot(); /* compute ue_bot of previous u_save */ | |
2243 curbuf->b_u_curhead = NULL; | |
2244 } | |
2245 } | |
2246 | |
2247 /* | |
772 | 2248 * ":undolist": List the leafs of the undo tree |
2249 */ | |
2250 void | |
2251 ex_undolist(eap) | |
2154
7c8c7c95a865
First step in the Vim 7.3 branch. Changed version numbers.
Bram Moolenaar <bram@zimbu.org>
parents:
1534
diff
changeset
|
2252 exarg_T *eap UNUSED; |
772 | 2253 { |
2254 garray_T ga; | |
2255 u_header_T *uhp; | |
2256 int mark; | |
2257 int nomark; | |
2258 int changes = 1; | |
2259 int i; | |
2260 | |
2261 /* | |
2262 * 1: walk the tree to find all leafs, put the info in "ga". | |
2263 * 2: sort the lines | |
2264 * 3: display the list | |
2265 */ | |
2266 mark = ++lastmark; | |
2267 nomark = ++lastmark; | |
2268 ga_init2(&ga, (int)sizeof(char *), 20); | |
2269 | |
2270 uhp = curbuf->b_u_oldhead; | |
2271 while (uhp != NULL) | |
2272 { | |
777 | 2273 if (uhp->uh_prev == NULL && uhp->uh_walk != nomark |
2274 && uhp->uh_walk != mark) | |
772 | 2275 { |
2276 if (ga_grow(&ga, 1) == FAIL) | |
2277 break; | |
2278 vim_snprintf((char *)IObuff, IOSIZE, "%6ld %7ld ", | |
2279 uhp->uh_seq, changes); | |
2280 u_add_time(IObuff + STRLEN(IObuff), IOSIZE - STRLEN(IObuff), | |
2281 uhp->uh_time); | |
2282 ((char_u **)(ga.ga_data))[ga.ga_len++] = vim_strsave(IObuff); | |
2283 } | |
2284 | |
2285 uhp->uh_walk = mark; | |
2286 | |
2287 /* go down in the tree if we haven't been there */ | |
2288 if (uhp->uh_prev != NULL && uhp->uh_prev->uh_walk != nomark | |
2289 && uhp->uh_prev->uh_walk != mark) | |
2290 { | |
2291 uhp = uhp->uh_prev; | |
2292 ++changes; | |
2293 } | |
2294 | |
2295 /* go to alternate branch if we haven't been there */ | |
2296 else if (uhp->uh_alt_next != NULL | |
2297 && uhp->uh_alt_next->uh_walk != nomark | |
2298 && uhp->uh_alt_next->uh_walk != mark) | |
2299 uhp = uhp->uh_alt_next; | |
2300 | |
2301 /* go up in the tree if we haven't been there and we are at the | |
2302 * start of alternate branches */ | |
2303 else if (uhp->uh_next != NULL && uhp->uh_alt_prev == NULL | |
2304 && uhp->uh_next->uh_walk != nomark | |
2305 && uhp->uh_next->uh_walk != mark) | |
2306 { | |
2307 uhp = uhp->uh_next; | |
2308 --changes; | |
2309 } | |
2310 | |
2311 else | |
2312 { | |
2313 /* need to backtrack; mark this node as done */ | |
2314 uhp->uh_walk = nomark; | |
2315 if (uhp->uh_alt_prev != NULL) | |
2316 uhp = uhp->uh_alt_prev; | |
2317 else | |
2318 { | |
2319 uhp = uhp->uh_next; | |
2320 --changes; | |
2321 } | |
2322 } | |
2323 } | |
2324 | |
2325 if (ga.ga_len == 0) | |
2326 MSG(_("Nothing to undo")); | |
2327 else | |
2328 { | |
2329 sort_strings((char_u **)ga.ga_data, ga.ga_len); | |
2330 | |
2331 msg_start(); | |
2332 msg_puts_attr((char_u *)_("number changes time"), hl_attr(HLF_T)); | |
2333 for (i = 0; i < ga.ga_len && !got_int; ++i) | |
2334 { | |
2335 msg_putchar('\n'); | |
2336 if (got_int) | |
2337 break; | |
2338 msg_puts(((char_u **)ga.ga_data)[i]); | |
2339 } | |
2340 msg_end(); | |
2341 | |
2342 ga_clear_strings(&ga); | |
2343 } | |
2344 } | |
2345 | |
2346 /* | |
2347 * Put the timestamp of an undo header in "buf[buflen]" in a nice format. | |
2348 */ | |
2349 static void | |
2350 u_add_time(buf, buflen, tt) | |
2351 char_u *buf; | |
2352 size_t buflen; | |
2353 time_t tt; | |
2354 { | |
2355 #ifdef HAVE_STRFTIME | |
2356 struct tm *curtime; | |
2357 | |
2358 if (time(NULL) - tt >= 100) | |
2359 { | |
2360 curtime = localtime(&tt); | |
810 | 2361 (void)strftime((char *)buf, buflen, "%H:%M:%S", curtime); |
772 | 2362 } |
2363 else | |
2364 #endif | |
835 | 2365 vim_snprintf((char *)buf, buflen, _("%ld seconds ago"), |
772 | 2366 (long)(time(NULL) - tt)); |
2367 } | |
2368 | |
2369 /* | |
697 | 2370 * ":undojoin": continue adding to the last entry list |
2371 */ | |
2372 void | |
2373 ex_undojoin(eap) | |
2154
7c8c7c95a865
First step in the Vim 7.3 branch. Changed version numbers.
Bram Moolenaar <bram@zimbu.org>
parents:
1534
diff
changeset
|
2374 exarg_T *eap UNUSED; |
697 | 2375 { |
839 | 2376 if (curbuf->b_u_newhead == NULL) |
2377 return; /* nothing changed before */ | |
2378 if (curbuf->b_u_curhead != NULL) | |
2379 { | |
2380 EMSG(_("E790: undojoin is not allowed after undo")); | |
2381 return; | |
2382 } | |
697 | 2383 if (!curbuf->b_u_synced) |
2384 return; /* already unsynced */ | |
2385 if (p_ul < 0) | |
2386 return; /* no entries, nothing to do */ | |
2387 else | |
2388 { | |
2389 /* Go back to the last entry */ | |
2390 curbuf->b_u_curhead = curbuf->b_u_newhead; | |
2391 curbuf->b_u_synced = FALSE; /* no entries, nothing to do */ | |
2392 } | |
2393 } | |
2394 | |
2395 /* | |
7 | 2396 * Called after writing the file and setting b_changed to FALSE. |
2397 * Now an undo means that the buffer is modified. | |
2398 */ | |
2399 void | |
2400 u_unchanged(buf) | |
2401 buf_T *buf; | |
2402 { | |
758 | 2403 u_unch_branch(buf->b_u_oldhead); |
2404 buf->b_did_warn = FALSE; | |
2405 } | |
2406 | |
2407 static void | |
2408 u_unch_branch(uhp) | |
2409 u_header_T *uhp; | |
2410 { | |
753 | 2411 u_header_T *uh; |
7 | 2412 |
758 | 2413 for (uh = uhp; uh != NULL; uh = uh->uh_prev) |
2414 { | |
7 | 2415 uh->uh_flags |= UH_CHANGED; |
758 | 2416 if (uh->uh_alt_next != NULL) |
2417 u_unch_branch(uh->uh_alt_next); /* recursive */ | |
2418 } | |
7 | 2419 } |
2420 | |
2421 /* | |
2422 * Get pointer to last added entry. | |
2423 * If it's not valid, give an error message and return NULL. | |
2424 */ | |
2425 static u_entry_T * | |
2426 u_get_headentry() | |
2427 { | |
2428 if (curbuf->b_u_newhead == NULL || curbuf->b_u_newhead->uh_entry == NULL) | |
2429 { | |
2430 EMSG(_("E439: undo list corrupt")); | |
2431 return NULL; | |
2432 } | |
2433 return curbuf->b_u_newhead->uh_entry; | |
2434 } | |
2435 | |
2436 /* | |
2437 * u_getbot(): compute the line number of the previous u_save | |
2438 * It is called only when b_u_synced is FALSE. | |
2439 */ | |
2440 static void | |
2441 u_getbot() | |
2442 { | |
2443 u_entry_T *uep; | |
2444 linenr_T extra; | |
2445 | |
2446 uep = u_get_headentry(); /* check for corrupt undo list */ | |
2447 if (uep == NULL) | |
2448 return; | |
2449 | |
2450 uep = curbuf->b_u_newhead->uh_getbot_entry; | |
2451 if (uep != NULL) | |
2452 { | |
2453 /* | |
2454 * the new ue_bot is computed from the number of lines that has been | |
2455 * inserted (0 - deleted) since calling u_save. This is equal to the | |
2456 * old line count subtracted from the current line count. | |
2457 */ | |
2458 extra = curbuf->b_ml.ml_line_count - uep->ue_lcount; | |
2459 uep->ue_bot = uep->ue_top + uep->ue_size + 1 + extra; | |
2460 if (uep->ue_bot < 1 || uep->ue_bot > curbuf->b_ml.ml_line_count) | |
2461 { | |
2462 EMSG(_("E440: undo line missing")); | |
2463 uep->ue_bot = uep->ue_top + 1; /* assume all lines deleted, will | |
2464 * get all the old lines back | |
2465 * without deleting the current | |
2466 * ones */ | |
2467 } | |
2468 | |
2469 curbuf->b_u_newhead->uh_getbot_entry = NULL; | |
2470 } | |
2471 | |
2472 curbuf->b_u_synced = TRUE; | |
2473 } | |
2474 | |
2475 /* | |
1415 | 2476 * Free one header "uhp" and its entry list and adjust the pointers. |
7 | 2477 */ |
2478 static void | |
758 | 2479 u_freeheader(buf, uhp, uhpp) |
168 | 2480 buf_T *buf; |
753 | 2481 u_header_T *uhp; |
2482 u_header_T **uhpp; /* if not NULL reset when freeing this header */ | |
7 | 2483 { |
1415 | 2484 u_header_T *uhap; |
2485 | |
753 | 2486 /* When there is an alternate redo list free that branch completely, |
2487 * because we can never go there. */ | |
2488 if (uhp->uh_alt_next != NULL) | |
2489 u_freebranch(buf, uhp->uh_alt_next, uhpp); | |
7 | 2490 |
753 | 2491 if (uhp->uh_alt_prev != NULL) |
2492 uhp->uh_alt_prev->uh_alt_next = NULL; | |
7 | 2493 |
753 | 2494 /* Update the links in the list to remove the header. */ |
7 | 2495 if (uhp->uh_next == NULL) |
168 | 2496 buf->b_u_oldhead = uhp->uh_prev; |
7 | 2497 else |
2498 uhp->uh_next->uh_prev = uhp->uh_prev; | |
2499 | |
2500 if (uhp->uh_prev == NULL) | |
168 | 2501 buf->b_u_newhead = uhp->uh_next; |
7 | 2502 else |
1415 | 2503 for (uhap = uhp->uh_prev; uhap != NULL; uhap = uhap->uh_alt_next) |
2504 uhap->uh_next = uhp->uh_next; | |
7 | 2505 |
753 | 2506 u_freeentries(buf, uhp, uhpp); |
2507 } | |
2508 | |
2509 /* | |
758 | 2510 * Free an alternate branch and any following alternate branches. |
753 | 2511 */ |
2512 static void | |
2513 u_freebranch(buf, uhp, uhpp) | |
2514 buf_T *buf; | |
2515 u_header_T *uhp; | |
2516 u_header_T **uhpp; /* if not NULL reset when freeing this header */ | |
2517 { | |
2518 u_header_T *tofree, *next; | |
2519 | |
1440 | 2520 /* If this is the top branch we may need to use u_freeheader() to update |
2521 * all the pointers. */ | |
2522 if (uhp == buf->b_u_oldhead) | |
2523 { | |
2524 u_freeheader(buf, uhp, uhpp); | |
2525 return; | |
2526 } | |
2527 | |
753 | 2528 if (uhp->uh_alt_prev != NULL) |
2529 uhp->uh_alt_prev->uh_alt_next = NULL; | |
2530 | |
2531 next = uhp; | |
2532 while (next != NULL) | |
2533 { | |
2534 tofree = next; | |
2535 if (tofree->uh_alt_next != NULL) | |
2536 u_freebranch(buf, tofree->uh_alt_next, uhpp); /* recursive */ | |
2537 next = tofree->uh_prev; | |
2538 u_freeentries(buf, tofree, uhpp); | |
2539 } | |
2540 } | |
2541 | |
2542 /* | |
2543 * Free all the undo entries for one header and the header itself. | |
2544 * This means that "uhp" is invalid when returning. | |
2545 */ | |
2546 static void | |
2547 u_freeentries(buf, uhp, uhpp) | |
2548 buf_T *buf; | |
2549 u_header_T *uhp; | |
2550 u_header_T **uhpp; /* if not NULL reset when freeing this header */ | |
2551 { | |
2552 u_entry_T *uep, *nuep; | |
2553 | |
2554 /* Check for pointers to the header that become invalid now. */ | |
2555 if (buf->b_u_curhead == uhp) | |
2556 buf->b_u_curhead = NULL; | |
1415 | 2557 if (buf->b_u_newhead == uhp) |
2558 buf->b_u_newhead = NULL; /* freeing the newest entry */ | |
753 | 2559 if (uhpp != NULL && uhp == *uhpp) |
2560 *uhpp = NULL; | |
2561 | |
2562 for (uep = uhp->uh_entry; uep != NULL; uep = nuep) | |
2563 { | |
2564 nuep = uep->ue_next; | |
2565 u_freeentry(uep, uep->ue_size); | |
2566 } | |
2567 | |
1415 | 2568 #ifdef U_DEBUG |
2569 uhp->uh_magic = 0; | |
2570 #endif | |
168 | 2571 U_FREE_LINE((char_u *)uhp); |
2572 --buf->b_u_numhead; | |
7 | 2573 } |
2574 | |
2575 /* | |
2576 * free entry 'uep' and 'n' lines in uep->ue_array[] | |
2577 */ | |
2578 static void | |
2579 u_freeentry(uep, n) | |
2580 u_entry_T *uep; | |
2581 long n; | |
2582 { | |
414 | 2583 while (n > 0) |
168 | 2584 U_FREE_LINE(uep->ue_array[--n]); |
359 | 2585 U_FREE_LINE((char_u *)uep->ue_array); |
1415 | 2586 #ifdef U_DEBUG |
2587 uep->ue_magic = 0; | |
2588 #endif | |
168 | 2589 U_FREE_LINE((char_u *)uep); |
7 | 2590 } |
2591 | |
2592 /* | |
2593 * invalidate the undo buffer; called when storage has already been released | |
2594 */ | |
2595 void | |
2596 u_clearall(buf) | |
2597 buf_T *buf; | |
2598 { | |
2599 buf->b_u_newhead = buf->b_u_oldhead = buf->b_u_curhead = NULL; | |
2600 buf->b_u_synced = TRUE; | |
2601 buf->b_u_numhead = 0; | |
2602 buf->b_u_line_ptr = NULL; | |
2603 buf->b_u_line_lnum = 0; | |
2604 } | |
2605 | |
2606 /* | |
2607 * save the line "lnum" for the "U" command | |
2608 */ | |
2609 void | |
2610 u_saveline(lnum) | |
2611 linenr_T lnum; | |
2612 { | |
2613 if (lnum == curbuf->b_u_line_lnum) /* line is already saved */ | |
2614 return; | |
356 | 2615 if (lnum < 1 || lnum > curbuf->b_ml.ml_line_count) /* should never happen */ |
7 | 2616 return; |
2617 u_clearline(); | |
2618 curbuf->b_u_line_lnum = lnum; | |
2619 if (curwin->w_cursor.lnum == lnum) | |
2620 curbuf->b_u_line_colnr = curwin->w_cursor.col; | |
2621 else | |
2622 curbuf->b_u_line_colnr = 0; | |
2623 if ((curbuf->b_u_line_ptr = u_save_line(lnum)) == NULL) | |
2624 do_outofmem_msg((long_u)0); | |
2625 } | |
2626 | |
2627 /* | |
2628 * clear the line saved for the "U" command | |
2629 * (this is used externally for crossing a line while in insert mode) | |
2630 */ | |
2631 void | |
2632 u_clearline() | |
2633 { | |
2634 if (curbuf->b_u_line_ptr != NULL) | |
2635 { | |
168 | 2636 U_FREE_LINE(curbuf->b_u_line_ptr); |
7 | 2637 curbuf->b_u_line_ptr = NULL; |
2638 curbuf->b_u_line_lnum = 0; | |
2639 } | |
2640 } | |
2641 | |
2642 /* | |
2643 * Implementation of the "U" command. | |
2644 * Differentiation from vi: "U" can be undone with the next "U". | |
2645 * We also allow the cursor to be in another line. | |
2646 */ | |
2647 void | |
2648 u_undoline() | |
2649 { | |
2650 colnr_T t; | |
2651 char_u *oldp; | |
2652 | |
2653 if (undo_off) | |
2654 return; | |
2655 | |
1534 | 2656 if (curbuf->b_u_line_ptr == NULL |
2657 || curbuf->b_u_line_lnum > curbuf->b_ml.ml_line_count) | |
7 | 2658 { |
2659 beep_flush(); | |
2660 return; | |
2661 } | |
1534 | 2662 |
2663 /* first save the line for the 'u' command */ | |
7 | 2664 if (u_savecommon(curbuf->b_u_line_lnum - 1, |
2665 curbuf->b_u_line_lnum + 1, (linenr_T)0) == FAIL) | |
2666 return; | |
2667 oldp = u_save_line(curbuf->b_u_line_lnum); | |
2668 if (oldp == NULL) | |
2669 { | |
2670 do_outofmem_msg((long_u)0); | |
2671 return; | |
2672 } | |
2673 ml_replace(curbuf->b_u_line_lnum, curbuf->b_u_line_ptr, TRUE); | |
2674 changed_bytes(curbuf->b_u_line_lnum, 0); | |
168 | 2675 U_FREE_LINE(curbuf->b_u_line_ptr); |
7 | 2676 curbuf->b_u_line_ptr = oldp; |
2677 | |
2678 t = curbuf->b_u_line_colnr; | |
2679 if (curwin->w_cursor.lnum == curbuf->b_u_line_lnum) | |
2680 curbuf->b_u_line_colnr = curwin->w_cursor.col; | |
2681 curwin->w_cursor.col = t; | |
2682 curwin->w_cursor.lnum = curbuf->b_u_line_lnum; | |
1534 | 2683 check_cursor_col(); |
7 | 2684 } |
2685 | |
2686 /* | |
168 | 2687 * There are two implementations of the memory management for undo: |
2688 * 1. Use the standard malloc()/free() functions. | |
2689 * This should be fast for allocating memory, but when a buffer is | |
2690 * abandoned every single allocated chunk must be freed, which may be slow. | |
2691 * 2. Allocate larger blocks of memory and keep track of chunks ourselves. | |
2692 * This is fast for abandoning, but the use of linked lists is slow for | |
2693 * finding a free chunk. Esp. when a lot of lines are changed or deleted. | |
2694 * A bit of profiling showed that the first method is faster, especially when | |
2695 * making a large number of changes, under the condition that malloc()/free() | |
2696 * is implemented efficiently. | |
2697 */ | |
2698 #ifdef U_USE_MALLOC | |
2699 /* | |
2700 * Version of undo memory allocation using malloc()/free() | |
2701 * | |
2702 * U_FREE_LINE() and U_ALLOC_LINE() are macros that invoke vim_free() and | |
2703 * lalloc() directly. | |
2704 */ | |
2705 | |
2706 /* | |
2707 * Free all allocated memory blocks for the buffer 'buf'. | |
2708 */ | |
2709 void | |
2710 u_blockfree(buf) | |
2711 buf_T *buf; | |
2712 { | |
753 | 2713 while (buf->b_u_oldhead != NULL) |
758 | 2714 u_freeheader(buf, buf->b_u_oldhead, NULL); |
359 | 2715 U_FREE_LINE(buf->b_u_line_ptr); |
168 | 2716 } |
2717 | |
2718 #else | |
2719 /* | |
2720 * Storage allocation for the undo lines and blocks of the current file. | |
2721 * Version where Vim keeps track of the available memory. | |
7 | 2722 */ |
2723 | |
2724 /* | |
2725 * Memory is allocated in relatively large blocks. These blocks are linked | |
2726 * in the allocated block list, headed by curbuf->b_block_head. They are all | |
2727 * freed when abandoning a file, so we don't have to free every single line. | |
2728 * The list is kept sorted on memory address. | |
2729 * block_alloc() allocates a block. | |
2730 * m_blockfree() frees all blocks. | |
2731 * | |
2732 * The available chunks of memory are kept in free chunk lists. There is | |
2733 * one free list for each block of allocated memory. The list is kept sorted | |
2734 * on memory address. | |
2735 * u_alloc_line() gets a chunk from the free lists. | |
2736 * u_free_line() returns a chunk to the free lists. | |
2737 * curbuf->b_m_search points to the chunk before the chunk that was | |
2738 * freed/allocated the last time. | |
2739 * curbuf->b_mb_current points to the b_head where curbuf->b_m_search | |
2740 * points into the free list. | |
2741 * | |
2742 * | |
2743 * b_block_head /---> block #1 /---> block #2 | |
2744 * mb_next ---/ mb_next ---/ mb_next ---> NULL | |
2745 * mb_info mb_info mb_info | |
2746 * | | | | |
2747 * V V V | |
2748 * NULL free chunk #1.1 free chunk #2.1 | |
2749 * | | | |
2750 * V V | |
2751 * free chunk #1.2 NULL | |
2752 * | | |
2753 * V | |
2754 * NULL | |
2755 * | |
2756 * When a single free chunk list would have been used, it could take a lot | |
2757 * of time in u_free_line() to find the correct place to insert a chunk in the | |
2758 * free list. The single free list would become very long when many lines are | |
2759 * changed (e.g. with :%s/^M$//). | |
2760 */ | |
2761 | |
2762 /* | |
2763 * this blocksize is used when allocating new lines | |
2764 */ | |
2765 #define MEMBLOCKSIZE 2044 | |
2766 | |
2767 /* | |
2768 * The size field contains the size of the chunk, including the size field | |
2769 * itself. | |
2770 * | |
2771 * When the chunk is not in-use it is preceded with the m_info structure. | |
2772 * The m_next field links it in one of the free chunk lists. | |
2773 * | |
2774 * On most unix systems structures have to be longword (32 or 64 bit) aligned. | |
2775 * On most other systems they are short (16 bit) aligned. | |
2776 */ | |
2777 | |
2778 /* the structure definitions are now in structs.h */ | |
2779 | |
2780 #ifdef ALIGN_LONG | |
2781 /* size of m_size */ | |
2782 # define M_OFFSET (sizeof(long_u)) | |
2783 #else | |
2784 /* size of m_size */ | |
2785 # define M_OFFSET (sizeof(short_u)) | |
2786 #endif | |
2787 | |
168 | 2788 static char_u *u_blockalloc __ARGS((long_u)); |
2789 | |
7 | 2790 /* |
2791 * Allocate a block of memory and link it in the allocated block list. | |
2792 */ | |
2793 static char_u * | |
2794 u_blockalloc(size) | |
2795 long_u size; | |
2796 { | |
2797 mblock_T *p; | |
2798 mblock_T *mp, *next; | |
2799 | |
2800 p = (mblock_T *)lalloc(size + sizeof(mblock_T), FALSE); | |
2801 if (p != NULL) | |
2802 { | |
2803 /* Insert the block into the allocated block list, keeping it | |
2804 sorted on address. */ | |
2805 for (mp = &curbuf->b_block_head; | |
2806 (next = mp->mb_next) != NULL && next < p; | |
2807 mp = next) | |
2808 ; | |
2809 p->mb_next = next; /* link in block list */ | |
2810 p->mb_size = size; | |
168 | 2811 p->mb_maxsize = 0; /* nothing free yet */ |
7 | 2812 mp->mb_next = p; |
2813 p->mb_info.m_next = NULL; /* clear free list */ | |
2814 p->mb_info.m_size = 0; | |
2815 curbuf->b_mb_current = p; /* remember current block */ | |
2816 curbuf->b_m_search = NULL; | |
2817 p++; /* return usable memory */ | |
2818 } | |
2819 return (char_u *)p; | |
2820 } | |
2821 | |
2822 /* | |
2823 * free all allocated memory blocks for the buffer 'buf' | |
2824 */ | |
2825 void | |
2826 u_blockfree(buf) | |
2827 buf_T *buf; | |
2828 { | |
2829 mblock_T *p, *np; | |
2830 | |
2831 for (p = buf->b_block_head.mb_next; p != NULL; p = np) | |
2832 { | |
2833 np = p->mb_next; | |
2834 vim_free(p); | |
2835 } | |
2836 buf->b_block_head.mb_next = NULL; | |
2837 buf->b_m_search = NULL; | |
2838 buf->b_mb_current = NULL; | |
2839 } | |
2840 | |
2841 /* | |
2842 * Free a chunk of memory for the current buffer. | |
2843 * Insert the chunk into the correct free list, keeping it sorted on address. | |
2844 */ | |
2845 static void | |
2846 u_free_line(ptr, keep) | |
2847 char_u *ptr; | |
2848 int keep; /* don't free the block when it's empty */ | |
2849 { | |
2850 minfo_T *next; | |
2851 minfo_T *prev, *curr; | |
2852 minfo_T *mp; | |
2853 mblock_T *nextb; | |
2854 mblock_T *prevb; | |
168 | 2855 long_u maxsize; |
7 | 2856 |
2857 if (ptr == NULL || ptr == IObuff) | |
2858 return; /* illegal address can happen in out-of-memory situations */ | |
2859 | |
2860 mp = (minfo_T *)(ptr - M_OFFSET); | |
2861 | |
2862 /* find block where chunk could be a part off */ | |
2863 /* if we change curbuf->b_mb_current, curbuf->b_m_search is set to NULL */ | |
2864 if (curbuf->b_mb_current == NULL || mp < (minfo_T *)curbuf->b_mb_current) | |
2865 { | |
2866 curbuf->b_mb_current = curbuf->b_block_head.mb_next; | |
2867 curbuf->b_m_search = NULL; | |
2868 } | |
2869 if ((nextb = curbuf->b_mb_current->mb_next) != NULL | |
2870 && (minfo_T *)nextb < mp) | |
2871 { | |
2872 curbuf->b_mb_current = nextb; | |
2873 curbuf->b_m_search = NULL; | |
2874 } | |
2875 while ((nextb = curbuf->b_mb_current->mb_next) != NULL | |
2876 && (minfo_T *)nextb < mp) | |
2877 curbuf->b_mb_current = nextb; | |
2878 | |
2879 curr = NULL; | |
2880 /* | |
2881 * If mp is smaller than curbuf->b_m_search->m_next go to the start of | |
2882 * the free list | |
2883 */ | |
2884 if (curbuf->b_m_search == NULL || mp < (curbuf->b_m_search->m_next)) | |
2885 next = &(curbuf->b_mb_current->mb_info); | |
2886 else | |
2887 next = curbuf->b_m_search; | |
2888 /* | |
2889 * The following loop is executed very often. | |
2890 * Therefore it has been optimized at the cost of readability. | |
2891 * Keep it fast! | |
2892 */ | |
2893 #ifdef SLOW_BUT_EASY_TO_READ | |
2894 do | |
2895 { | |
2896 prev = curr; | |
2897 curr = next; | |
2898 next = next->m_next; | |
2899 } | |
2900 while (mp > next && next != NULL); | |
2901 #else | |
2902 do /* first, middle, last */ | |
2903 { | |
2904 prev = next->m_next; /* curr, next, prev */ | |
2905 if (prev == NULL || mp <= prev) | |
2906 { | |
2907 prev = curr; | |
2908 curr = next; | |
2909 next = next->m_next; | |
2910 break; | |
2911 } | |
2912 curr = prev->m_next; /* next, prev, curr */ | |
2913 if (curr == NULL || mp <= curr) | |
2914 { | |
2915 prev = next; | |
2916 curr = prev->m_next; | |
2917 next = curr->m_next; | |
2918 break; | |
2919 } | |
2920 next = curr->m_next; /* prev, curr, next */ | |
2921 } | |
2922 while (mp > next && next != NULL); | |
2923 #endif | |
2924 | |
2925 /* if *mp and *next are concatenated, join them into one chunk */ | |
2926 if ((char_u *)mp + mp->m_size == (char_u *)next) | |
2927 { | |
2928 mp->m_size += next->m_size; | |
2929 mp->m_next = next->m_next; | |
2930 } | |
2931 else | |
2932 mp->m_next = next; | |
168 | 2933 maxsize = mp->m_size; |
7 | 2934 |
2935 /* if *curr and *mp are concatenated, join them */ | |
2936 if (prev != NULL && (char_u *)curr + curr->m_size == (char_u *)mp) | |
2937 { | |
2938 curr->m_size += mp->m_size; | |
168 | 2939 maxsize = curr->m_size; |
7 | 2940 curr->m_next = mp->m_next; |
2941 curbuf->b_m_search = prev; | |
2942 } | |
2943 else | |
2944 { | |
2945 curr->m_next = mp; | |
2946 curbuf->b_m_search = curr; /* put curbuf->b_m_search before freed | |
2947 chunk */ | |
2948 } | |
2949 | |
2950 /* | |
1210 | 2951 * If the block only contains free memory now, release it. |
7 | 2952 */ |
2953 if (!keep && curbuf->b_mb_current->mb_size | |
2954 == curbuf->b_mb_current->mb_info.m_next->m_size) | |
2955 { | |
2956 /* Find the block before the current one to be able to unlink it from | |
2957 * the list of blocks. */ | |
2958 prevb = &curbuf->b_block_head; | |
2959 for (nextb = prevb->mb_next; nextb != curbuf->b_mb_current; | |
2960 nextb = nextb->mb_next) | |
2961 prevb = nextb; | |
2962 prevb->mb_next = nextb->mb_next; | |
2963 vim_free(nextb); | |
2964 curbuf->b_mb_current = NULL; | |
2965 curbuf->b_m_search = NULL; | |
2966 } | |
168 | 2967 else if (curbuf->b_mb_current->mb_maxsize < maxsize) |
2968 curbuf->b_mb_current->mb_maxsize = maxsize; | |
7 | 2969 } |
2970 | |
2971 /* | |
2972 * Allocate and initialize a new line structure with room for at least | |
2973 * 'size' characters plus a terminating NUL. | |
2974 */ | |
2975 static char_u * | |
2976 u_alloc_line(size) | |
2977 unsigned size; | |
2978 { | |
2979 minfo_T *mp, *mprev, *mp2; | |
2980 mblock_T *mbp; | |
2981 int size_align; | |
2982 | |
2983 /* | |
2984 * Add room for size field and trailing NUL byte. | |
2985 * Adjust for minimal size (must be able to store minfo_T | |
2986 * plus a trailing NUL, so the chunk can be released again) | |
2987 */ | |
2988 size += M_OFFSET + 1; | |
2989 if (size < sizeof(minfo_T) + 1) | |
2990 size = sizeof(minfo_T) + 1; | |
2991 | |
2992 /* | |
2993 * round size up for alignment | |
2994 */ | |
2995 size_align = (size + ALIGN_MASK) & ~ALIGN_MASK; | |
2996 | |
2997 /* | |
2998 * If curbuf->b_m_search is NULL (uninitialized free list) start at | |
2999 * curbuf->b_block_head | |
3000 */ | |
3001 if (curbuf->b_mb_current == NULL || curbuf->b_m_search == NULL) | |
3002 { | |
3003 curbuf->b_mb_current = &curbuf->b_block_head; | |
3004 curbuf->b_m_search = &(curbuf->b_block_head.mb_info); | |
3005 } | |
3006 | |
168 | 3007 /* Search for a block with enough space. */ |
7 | 3008 mbp = curbuf->b_mb_current; |
168 | 3009 while (mbp->mb_maxsize < size_align) |
7 | 3010 { |
168 | 3011 if (mbp->mb_next != NULL) |
7 | 3012 mbp = mbp->mb_next; |
3013 else | |
3014 mbp = &curbuf->b_block_head; | |
168 | 3015 if (mbp == curbuf->b_mb_current) |
7 | 3016 { |
168 | 3017 int n = (size_align > (MEMBLOCKSIZE / 4) |
3018 ? size_align : MEMBLOCKSIZE); | |
7 | 3019 |
168 | 3020 /* Back where we started in block list: need to add a new block |
3021 * with enough space. */ | |
3022 mp = (minfo_T *)u_blockalloc((long_u)n); | |
3023 if (mp == NULL) | |
3024 return (NULL); | |
3025 mp->m_size = n; | |
3026 u_free_line((char_u *)mp + M_OFFSET, TRUE); | |
3027 mbp = curbuf->b_mb_current; | |
3028 break; | |
3029 } | |
3030 } | |
3031 if (mbp != curbuf->b_mb_current) | |
3032 curbuf->b_m_search = &(mbp->mb_info); | |
3033 | |
3034 /* In this block find a chunk with enough space. */ | |
3035 mprev = curbuf->b_m_search; | |
3036 mp = curbuf->b_m_search->m_next; | |
407 | 3037 for (;;) |
168 | 3038 { |
3039 if (mp == NULL) /* at end of the list */ | |
3040 mp = &(mbp->mb_info); /* wrap around to begin */ | |
3041 if (mp->m_size >= size) | |
3042 break; | |
3043 if (mp == curbuf->b_m_search) | |
3044 { | |
3045 /* back where we started in free chunk list: "cannot happen" */ | |
3046 EMSG2(_(e_intern2), "u_alloc_line()"); | |
3047 return NULL; | |
7 | 3048 } |
3049 mprev = mp; | |
168 | 3050 mp = mp->m_next; |
7 | 3051 } |
3052 | |
168 | 3053 /* when using the largest chunk adjust mb_maxsize */ |
3054 if (mp->m_size >= mbp->mb_maxsize) | |
3055 mbp->mb_maxsize = 0; | |
3056 | |
7 | 3057 /* if the chunk we found is large enough, split it up in two */ |
3058 if ((long)mp->m_size - size_align >= (long)(sizeof(minfo_T) + 1)) | |
3059 { | |
3060 mp2 = (minfo_T *)((char_u *)mp + size_align); | |
3061 mp2->m_size = mp->m_size - size_align; | |
3062 mp2->m_next = mp->m_next; | |
3063 mprev->m_next = mp2; | |
3064 mp->m_size = size_align; | |
3065 } | |
3066 else /* remove *mp from the free list */ | |
3067 { | |
3068 mprev->m_next = mp->m_next; | |
3069 } | |
3070 curbuf->b_m_search = mprev; | |
3071 curbuf->b_mb_current = mbp; | |
3072 | |
168 | 3073 /* If using the largest chunk need to find the new largest chunk */ |
3074 if (mbp->mb_maxsize == 0) | |
3075 for (mp2 = &(mbp->mb_info); mp2 != NULL; mp2 = mp2->m_next) | |
3076 if (mbp->mb_maxsize < mp2->m_size) | |
3077 mbp->mb_maxsize = mp2->m_size; | |
3078 | |
7 | 3079 mp = (minfo_T *)((char_u *)mp + M_OFFSET); |
3080 *(char_u *)mp = NUL; /* set the first byte to NUL */ | |
3081 | |
3082 return ((char_u *)mp); | |
3083 } | |
168 | 3084 #endif |
7 | 3085 |
3086 /* | |
3087 * u_save_line(): allocate memory with u_alloc_line() and copy line 'lnum' | |
3088 * into it. | |
3089 */ | |
3090 static char_u * | |
3091 u_save_line(lnum) | |
3092 linenr_T lnum; | |
3093 { | |
3094 char_u *src; | |
3095 char_u *dst; | |
3096 unsigned len; | |
3097 | |
3098 src = ml_get(lnum); | |
3099 len = (unsigned)STRLEN(src); | |
168 | 3100 if ((dst = U_ALLOC_LINE(len)) != NULL) |
7 | 3101 mch_memmove(dst, src, (size_t)(len + 1)); |
3102 return (dst); | |
3103 } | |
3104 | |
3105 /* | |
3106 * Check if the 'modified' flag is set, or 'ff' has changed (only need to | |
3107 * check the first character, because it can only be "dos", "unix" or "mac"). | |
3108 * "nofile" and "scratch" type buffers are considered to always be unchanged. | |
3109 */ | |
3110 int | |
3111 bufIsChanged(buf) | |
3112 buf_T *buf; | |
3113 { | |
3114 return | |
3115 #ifdef FEAT_QUICKFIX | |
3116 !bt_dontwrite(buf) && | |
3117 #endif | |
3118 (buf->b_changed || file_ff_differs(buf)); | |
3119 } | |
3120 | |
3121 int | |
3122 curbufIsChanged() | |
3123 { | |
3124 return | |
3125 #ifdef FEAT_QUICKFIX | |
3126 !bt_dontwrite(curbuf) && | |
3127 #endif | |
3128 (curbuf->b_changed || file_ff_differs(curbuf)); | |
3129 } |