wgenops.c 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609
  1. /* Copyright (C) 1993-2026 Free Software Foundation, Inc.
  2. This file is part of the GNU C Library.
  3. The GNU C Library is free software; you can redistribute it and/or
  4. modify it under the terms of the GNU Lesser General Public
  5. License as published by the Free Software Foundation; either
  6. version 2.1 of the License, or (at your option) any later version.
  7. The GNU C Library is distributed in the hope that it will be useful,
  8. but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  10. Lesser General Public License for more details.
  11. You should have received a copy of the GNU Lesser General Public
  12. License along with the GNU C Library; if not, see
  13. <https://www.gnu.org/licenses/>.
  14. As a special exception, if you link the code in this file with
  15. files compiled with a GNU compiler to produce an executable,
  16. that does not cause the resulting executable to be covered by
  17. the GNU Lesser General Public License. This exception does not
  18. however invalidate any other reasons why the executable file
  19. might be covered by the GNU Lesser General Public License.
  20. This exception applies to code released by its copyright holders
  21. in files containing the exception. */
  22. /* Generic or default I/O operations. */
  23. #include "libioP.h"
  24. #include <stdlib.h>
  25. #include <string.h>
  26. #include <wchar.h>
  27. static int save_for_wbackup (FILE *fp, wchar_t *end_p) __THROW;
  28. /* Return minimum _pos markers
  29. Assumes the current get area is the main get area. */
  30. ssize_t
  31. _IO_least_wmarker (FILE *fp, wchar_t *end_p)
  32. {
  33. ssize_t least_so_far = end_p - fp->_wide_data->_IO_read_base;
  34. struct _IO_marker *mark;
  35. for (mark = fp->_markers; mark != NULL; mark = mark->_next)
  36. if (mark->_pos < least_so_far)
  37. least_so_far = mark->_pos;
  38. return least_so_far;
  39. }
  40. libc_hidden_def (_IO_least_wmarker)
  41. /* Switch current get area from backup buffer to (start of) main get area. */
  42. void
  43. _IO_switch_to_main_wget_area (FILE *fp)
  44. {
  45. wchar_t *tmp;
  46. fp->_flags &= ~_IO_IN_BACKUP;
  47. /* Swap _IO_read_end and _IO_save_end. */
  48. tmp = fp->_wide_data->_IO_read_end;
  49. fp->_wide_data->_IO_read_end = fp->_wide_data->_IO_save_end;
  50. fp->_wide_data->_IO_save_end= tmp;
  51. /* Swap _IO_read_base and _IO_save_base. */
  52. tmp = fp->_wide_data->_IO_read_base;
  53. fp->_wide_data->_IO_read_base = fp->_wide_data->_IO_save_base;
  54. fp->_wide_data->_IO_save_base = tmp;
  55. /* Set _IO_read_ptr. */
  56. fp->_wide_data->_IO_read_ptr = fp->_wide_data->_IO_read_base;
  57. }
  58. libc_hidden_def (_IO_switch_to_main_wget_area)
  59. /* Switch current get area from main get area to (end of) backup area. */
  60. void
  61. _IO_switch_to_wbackup_area (FILE *fp)
  62. {
  63. wchar_t *tmp;
  64. fp->_flags |= _IO_IN_BACKUP;
  65. /* Swap _IO_read_end and _IO_save_end. */
  66. tmp = fp->_wide_data->_IO_read_end;
  67. fp->_wide_data->_IO_read_end = fp->_wide_data->_IO_save_end;
  68. fp->_wide_data->_IO_save_end = tmp;
  69. /* Swap _IO_read_base and _IO_save_base. */
  70. tmp = fp->_wide_data->_IO_read_base;
  71. fp->_wide_data->_IO_read_base = fp->_wide_data->_IO_save_base;
  72. fp->_wide_data->_IO_save_base = tmp;
  73. /* Set _IO_read_ptr. */
  74. fp->_wide_data->_IO_read_ptr = fp->_wide_data->_IO_read_end;
  75. }
  76. libc_hidden_def (_IO_switch_to_wbackup_area)
  77. void
  78. _IO_wsetb (FILE *f, wchar_t *b, wchar_t *eb, int a)
  79. {
  80. if (f->_wide_data->_IO_buf_base && !(f->_flags2 & _IO_FLAGS2_USER_WBUF))
  81. free (f->_wide_data->_IO_buf_base);
  82. f->_wide_data->_IO_buf_base = b;
  83. f->_wide_data->_IO_buf_end = eb;
  84. if (a)
  85. f->_flags2 &= ~_IO_FLAGS2_USER_WBUF;
  86. else
  87. f->_flags2 |= _IO_FLAGS2_USER_WBUF;
  88. }
  89. libc_hidden_def (_IO_wsetb)
  90. wint_t
  91. _IO_wdefault_pbackfail (FILE *fp, wint_t c)
  92. {
  93. if (fp->_wide_data->_IO_read_ptr > fp->_wide_data->_IO_read_base
  94. && !_IO_in_backup (fp)
  95. && (wint_t) fp->_IO_read_ptr[-1] == c)
  96. --fp->_IO_read_ptr;
  97. else
  98. {
  99. /* Need to handle a filebuf in write mode (switch to read mode). FIXME!*/
  100. if (!_IO_in_backup (fp))
  101. {
  102. /* We need to keep the invariant that the main get area
  103. logically follows the backup area. */
  104. if (fp->_wide_data->_IO_read_ptr > fp->_wide_data->_IO_read_base
  105. && _IO_have_wbackup (fp))
  106. {
  107. if (save_for_wbackup (fp, fp->_wide_data->_IO_read_ptr))
  108. return WEOF;
  109. }
  110. else if (!_IO_have_wbackup (fp))
  111. {
  112. /* No backup buffer: allocate one. */
  113. /* Use nshort buffer, if unused? (probably not) FIXME */
  114. int backup_size = 128;
  115. wchar_t *bbuf = (wchar_t *) malloc (backup_size
  116. * sizeof (wchar_t));
  117. if (bbuf == NULL)
  118. return WEOF;
  119. fp->_wide_data->_IO_save_base = bbuf;
  120. fp->_wide_data->_IO_save_end = (fp->_wide_data->_IO_save_base
  121. + backup_size);
  122. fp->_wide_data->_IO_backup_base = fp->_wide_data->_IO_save_end;
  123. }
  124. fp->_wide_data->_IO_read_base = fp->_wide_data->_IO_read_ptr;
  125. _IO_switch_to_wbackup_area (fp);
  126. }
  127. else if (fp->_wide_data->_IO_read_ptr <= fp->_wide_data->_IO_read_base)
  128. {
  129. /* Increase size of existing backup buffer. */
  130. size_t new_size;
  131. size_t old_size = (fp->_wide_data->_IO_read_end
  132. - fp->_wide_data->_IO_read_base);
  133. wchar_t *new_buf;
  134. new_size = 2 * old_size;
  135. new_buf = (wchar_t *) malloc (new_size * sizeof (wchar_t));
  136. if (new_buf == NULL)
  137. return WEOF;
  138. __wmemcpy (new_buf + (new_size - old_size),
  139. fp->_wide_data->_IO_read_base, old_size);
  140. free (fp->_wide_data->_IO_read_base);
  141. _IO_wsetg (fp, new_buf, new_buf + (new_size - old_size),
  142. new_buf + new_size);
  143. fp->_wide_data->_IO_backup_base = fp->_wide_data->_IO_read_ptr;
  144. }
  145. *--fp->_wide_data->_IO_read_ptr = c;
  146. }
  147. return c;
  148. }
  149. libc_hidden_def (_IO_wdefault_pbackfail)
  150. void
  151. _IO_wdefault_finish (FILE *fp, int dummy)
  152. {
  153. struct _IO_marker *mark;
  154. if (fp->_wide_data->_IO_buf_base && !(fp->_flags2 & _IO_FLAGS2_USER_WBUF))
  155. {
  156. free (fp->_wide_data->_IO_buf_base);
  157. fp->_wide_data->_IO_buf_base = fp->_wide_data->_IO_buf_end = NULL;
  158. }
  159. for (mark = fp->_markers; mark != NULL; mark = mark->_next)
  160. mark->_sbuf = NULL;
  161. if (fp->_IO_save_base)
  162. {
  163. free (fp->_wide_data->_IO_save_base);
  164. fp->_IO_save_base = NULL;
  165. }
  166. #ifdef _IO_MTSAFE_IO
  167. if (fp->_lock != NULL)
  168. _IO_lock_fini (*fp->_lock);
  169. #endif
  170. _IO_un_link ((struct _IO_FILE_plus *) fp);
  171. }
  172. libc_hidden_def (_IO_wdefault_finish)
  173. wint_t
  174. _IO_wdefault_uflow (FILE *fp)
  175. {
  176. wint_t wch;
  177. wch = _IO_UNDERFLOW (fp);
  178. if (wch == WEOF)
  179. return WEOF;
  180. return *fp->_wide_data->_IO_read_ptr++;
  181. }
  182. libc_hidden_def (_IO_wdefault_uflow)
  183. wint_t
  184. __woverflow (FILE *f, wint_t wch)
  185. {
  186. if (f->_mode == 0)
  187. _IO_fwide (f, 1);
  188. return _IO_OVERFLOW (f, wch);
  189. }
  190. libc_hidden_def (__woverflow)
  191. wint_t
  192. __wuflow (FILE *fp)
  193. {
  194. if (fp->_mode < 0 || (fp->_mode == 0 && _IO_fwide (fp, 1) != 1))
  195. return WEOF;
  196. if (fp->_mode == 0)
  197. _IO_fwide (fp, 1);
  198. if (_IO_in_put_mode (fp))
  199. if (_IO_switch_to_wget_mode (fp) == EOF)
  200. return WEOF;
  201. if (fp->_wide_data->_IO_read_ptr < fp->_wide_data->_IO_read_end)
  202. return *fp->_wide_data->_IO_read_ptr++;
  203. if (_IO_in_backup (fp))
  204. {
  205. _IO_switch_to_main_wget_area (fp);
  206. if (fp->_wide_data->_IO_read_ptr < fp->_wide_data->_IO_read_end)
  207. return *fp->_wide_data->_IO_read_ptr++;
  208. }
  209. if (_IO_have_markers (fp))
  210. {
  211. if (save_for_wbackup (fp, fp->_wide_data->_IO_read_end))
  212. return WEOF;
  213. }
  214. else if (_IO_have_wbackup (fp))
  215. _IO_free_wbackup_area (fp);
  216. return _IO_UFLOW (fp);
  217. }
  218. libc_hidden_def (__wuflow)
  219. wint_t
  220. __wunderflow (FILE *fp)
  221. {
  222. if (fp->_mode < 0 || (fp->_mode == 0 && _IO_fwide (fp, 1) != 1))
  223. return WEOF;
  224. if (fp->_mode == 0)
  225. _IO_fwide (fp, 1);
  226. if (_IO_in_put_mode (fp))
  227. if (_IO_switch_to_wget_mode (fp) == EOF)
  228. return WEOF;
  229. if (fp->_wide_data->_IO_read_ptr < fp->_wide_data->_IO_read_end)
  230. return *fp->_wide_data->_IO_read_ptr;
  231. if (_IO_in_backup (fp))
  232. {
  233. _IO_switch_to_main_wget_area (fp);
  234. if (fp->_wide_data->_IO_read_ptr < fp->_wide_data->_IO_read_end)
  235. return *fp->_wide_data->_IO_read_ptr;
  236. }
  237. if (_IO_have_markers (fp))
  238. {
  239. if (save_for_wbackup (fp, fp->_wide_data->_IO_read_end))
  240. return WEOF;
  241. }
  242. else if (_IO_have_backup (fp))
  243. _IO_free_wbackup_area (fp);
  244. return _IO_UNDERFLOW (fp);
  245. }
  246. libc_hidden_def (__wunderflow)
  247. size_t
  248. _IO_wdefault_xsputn (FILE *f, const void *data, size_t n)
  249. {
  250. const wchar_t *s = (const wchar_t *) data;
  251. size_t more = n;
  252. if (more <= 0)
  253. return 0;
  254. for (;;)
  255. {
  256. /* Space available. */
  257. ssize_t count = (f->_wide_data->_IO_write_end
  258. - f->_wide_data->_IO_write_ptr);
  259. if (count > 0)
  260. {
  261. if ((size_t) count > more)
  262. count = more;
  263. if (count > 20)
  264. {
  265. f->_wide_data->_IO_write_ptr =
  266. __wmempcpy (f->_wide_data->_IO_write_ptr, s, count);
  267. s += count;
  268. }
  269. else if (count <= 0)
  270. count = 0;
  271. else
  272. {
  273. wchar_t *p = f->_wide_data->_IO_write_ptr;
  274. ssize_t i;
  275. for (i = count; --i >= 0; )
  276. *p++ = *s++;
  277. f->_wide_data->_IO_write_ptr = p;
  278. }
  279. more -= count;
  280. }
  281. if (more == 0 || __woverflow (f, *s++) == WEOF)
  282. break;
  283. more--;
  284. }
  285. return n - more;
  286. }
  287. libc_hidden_def (_IO_wdefault_xsputn)
  288. size_t
  289. _IO_wdefault_xsgetn (FILE *fp, void *data, size_t n)
  290. {
  291. size_t more = n;
  292. wchar_t *s = (wchar_t*) data;
  293. for (;;)
  294. {
  295. /* Data available. */
  296. ssize_t count = (fp->_wide_data->_IO_read_end
  297. - fp->_wide_data->_IO_read_ptr);
  298. if (count > 0)
  299. {
  300. if ((size_t) count > more)
  301. count = more;
  302. if (count > 20)
  303. {
  304. s = __wmempcpy (s, fp->_wide_data->_IO_read_ptr, count);
  305. fp->_wide_data->_IO_read_ptr += count;
  306. }
  307. else if (count <= 0)
  308. count = 0;
  309. else
  310. {
  311. wchar_t *p = fp->_wide_data->_IO_read_ptr;
  312. int i = (int) count;
  313. while (--i >= 0)
  314. *s++ = *p++;
  315. fp->_wide_data->_IO_read_ptr = p;
  316. }
  317. more -= count;
  318. }
  319. if (more == 0 || __wunderflow (fp) == WEOF)
  320. break;
  321. }
  322. return n - more;
  323. }
  324. libc_hidden_def (_IO_wdefault_xsgetn)
  325. void
  326. _IO_wdoallocbuf (FILE *fp)
  327. {
  328. if (fp->_wide_data->_IO_buf_base)
  329. return;
  330. if (!(fp->_flags & _IO_UNBUFFERED))
  331. if ((wint_t)_IO_WDOALLOCATE (fp) != WEOF)
  332. return;
  333. _IO_wsetb (fp, fp->_wide_data->_shortbuf,
  334. fp->_wide_data->_shortbuf + 1, 0);
  335. }
  336. libc_hidden_def (_IO_wdoallocbuf)
  337. int
  338. _IO_wdefault_doallocate (FILE *fp)
  339. {
  340. wchar_t *buf = (wchar_t *)malloc (BUFSIZ);
  341. if (__glibc_unlikely (buf == NULL))
  342. return EOF;
  343. _IO_wsetb (fp, buf, buf + BUFSIZ / sizeof *buf, 1);
  344. return 1;
  345. }
  346. libc_hidden_def (_IO_wdefault_doallocate)
  347. int
  348. _IO_switch_to_wget_mode (FILE *fp)
  349. {
  350. if (fp->_wide_data->_IO_write_ptr > fp->_wide_data->_IO_write_base)
  351. if ((wint_t)_IO_WOVERFLOW (fp, WEOF) == WEOF)
  352. return EOF;
  353. if (_IO_in_backup (fp))
  354. fp->_wide_data->_IO_read_base = fp->_wide_data->_IO_backup_base;
  355. else
  356. {
  357. fp->_wide_data->_IO_read_base = fp->_wide_data->_IO_buf_base;
  358. if (fp->_wide_data->_IO_write_ptr > fp->_wide_data->_IO_read_end)
  359. fp->_wide_data->_IO_read_end = fp->_wide_data->_IO_write_ptr;
  360. }
  361. fp->_wide_data->_IO_read_ptr = fp->_wide_data->_IO_write_ptr;
  362. fp->_wide_data->_IO_write_base = fp->_wide_data->_IO_write_ptr
  363. = fp->_wide_data->_IO_write_end = fp->_wide_data->_IO_read_ptr;
  364. fp->_flags &= ~_IO_CURRENTLY_PUTTING;
  365. return 0;
  366. }
  367. libc_hidden_def (_IO_switch_to_wget_mode)
  368. void
  369. _IO_free_wbackup_area (FILE *fp)
  370. {
  371. if (_IO_in_backup (fp))
  372. _IO_switch_to_main_wget_area (fp); /* Just in case. */
  373. free (fp->_wide_data->_IO_save_base);
  374. fp->_wide_data->_IO_save_base = NULL;
  375. fp->_wide_data->_IO_save_end = NULL;
  376. fp->_wide_data->_IO_backup_base = NULL;
  377. }
  378. libc_hidden_def (_IO_free_wbackup_area)
  379. static int
  380. save_for_wbackup (FILE *fp, wchar_t *end_p)
  381. {
  382. /* Append [_IO_read_base..end_p] to backup area. */
  383. ssize_t least_mark = _IO_least_wmarker (fp, end_p);
  384. /* needed_size is how much space we need in the backup area. */
  385. size_t needed_size = ((end_p - fp->_wide_data->_IO_read_base)
  386. - least_mark);
  387. /* FIXME: Dubious arithmetic if pointers are NULL */
  388. size_t current_Bsize = (fp->_wide_data->_IO_save_end
  389. - fp->_wide_data->_IO_save_base);
  390. size_t avail; /* Extra space available for future expansion. */
  391. ssize_t delta;
  392. struct _IO_marker *mark;
  393. if (needed_size > current_Bsize)
  394. {
  395. wchar_t *new_buffer;
  396. avail = 100;
  397. new_buffer = (wchar_t *) malloc ((avail + needed_size)
  398. * sizeof (wchar_t));
  399. if (new_buffer == NULL)
  400. return EOF; /* FIXME */
  401. if (least_mark < 0)
  402. {
  403. __wmempcpy (__wmempcpy (new_buffer + avail,
  404. fp->_wide_data->_IO_save_end + least_mark,
  405. -least_mark),
  406. fp->_wide_data->_IO_read_base,
  407. end_p - fp->_wide_data->_IO_read_base);
  408. }
  409. else
  410. {
  411. __wmemcpy (new_buffer + avail,
  412. fp->_wide_data->_IO_read_base + least_mark,
  413. needed_size);
  414. }
  415. free (fp->_wide_data->_IO_save_base);
  416. fp->_wide_data->_IO_save_base = new_buffer;
  417. fp->_wide_data->_IO_save_end = new_buffer + avail + needed_size;
  418. }
  419. else
  420. {
  421. avail = current_Bsize - needed_size;
  422. if (least_mark < 0)
  423. {
  424. __wmemmove (fp->_wide_data->_IO_save_base + avail,
  425. fp->_wide_data->_IO_save_end + least_mark,
  426. -least_mark);
  427. __wmemcpy (fp->_wide_data->_IO_save_base + avail - least_mark,
  428. fp->_wide_data->_IO_read_base,
  429. end_p - fp->_wide_data->_IO_read_base);
  430. }
  431. else if (needed_size > 0)
  432. __wmemcpy (fp->_wide_data->_IO_save_base + avail,
  433. fp->_wide_data->_IO_read_base + least_mark,
  434. needed_size);
  435. }
  436. fp->_wide_data->_IO_backup_base = fp->_wide_data->_IO_save_base + avail;
  437. /* Adjust all the streammarkers. */
  438. delta = end_p - fp->_wide_data->_IO_read_base;
  439. for (mark = fp->_markers; mark != NULL; mark = mark->_next)
  440. mark->_pos -= delta;
  441. return 0;
  442. }
  443. wint_t
  444. _IO_sputbackwc (FILE *fp, wint_t c)
  445. {
  446. wint_t result;
  447. if (fp->_wide_data->_IO_read_ptr > fp->_wide_data->_IO_read_base
  448. && (wchar_t)fp->_wide_data->_IO_read_ptr[-1] == (wchar_t) c)
  449. {
  450. fp->_wide_data->_IO_read_ptr--;
  451. result = c;
  452. }
  453. else
  454. result = _IO_PBACKFAIL (fp, c);
  455. if (result != WEOF)
  456. fp->_flags &= ~_IO_EOF_SEEN;
  457. return result;
  458. }
  459. libc_hidden_def (_IO_sputbackwc)
  460. wint_t
  461. _IO_sungetwc (FILE *fp)
  462. {
  463. wint_t result;
  464. if (fp->_wide_data->_IO_read_ptr > fp->_wide_data->_IO_read_base)
  465. {
  466. fp->_wide_data->_IO_read_ptr--;
  467. result = *fp->_wide_data->_IO_read_ptr;
  468. }
  469. else
  470. result = _IO_PBACKFAIL (fp, EOF);
  471. if (result != WEOF)
  472. fp->_flags &= ~_IO_EOF_SEEN;
  473. return result;
  474. }
  475. unsigned
  476. _IO_adjust_wcolumn (unsigned start, const wchar_t *line, int count)
  477. {
  478. const wchar_t *ptr = line + count;
  479. while (ptr > line)
  480. if (*--ptr == L'\n')
  481. return line + count - ptr - 1;
  482. return start + count;
  483. }
  484. void
  485. _IO_init_wmarker (struct _IO_marker *marker, FILE *fp)
  486. {
  487. marker->_sbuf = fp;
  488. if (_IO_in_put_mode (fp))
  489. _IO_switch_to_wget_mode (fp);
  490. if (_IO_in_backup (fp))
  491. marker->_pos = fp->_wide_data->_IO_read_ptr - fp->_wide_data->_IO_read_end;
  492. else
  493. marker->_pos = (fp->_wide_data->_IO_read_ptr
  494. - fp->_wide_data->_IO_read_base);
  495. /* Should perhaps sort the chain? */
  496. marker->_next = fp->_markers;
  497. fp->_markers = marker;
  498. }
  499. #define BAD_DELTA EOF
  500. /* Return difference between MARK and current position of MARK's stream. */
  501. int
  502. _IO_wmarker_delta (struct _IO_marker *mark)
  503. {
  504. int cur_pos;
  505. if (mark->_sbuf == NULL)
  506. return BAD_DELTA;
  507. if (_IO_in_backup (mark->_sbuf))
  508. cur_pos = (mark->_sbuf->_wide_data->_IO_read_ptr
  509. - mark->_sbuf->_wide_data->_IO_read_end);
  510. else
  511. cur_pos = (mark->_sbuf->_wide_data->_IO_read_ptr
  512. - mark->_sbuf->_wide_data->_IO_read_base);
  513. return mark->_pos - cur_pos;
  514. }
  515. int
  516. _IO_seekwmark (FILE *fp, struct _IO_marker *mark, int delta)
  517. {
  518. if (mark->_sbuf != fp)
  519. return EOF;
  520. if (mark->_pos >= 0)
  521. {
  522. if (_IO_in_backup (fp))
  523. _IO_switch_to_main_wget_area (fp);
  524. fp->_wide_data->_IO_read_ptr = (fp->_wide_data->_IO_read_base
  525. + mark->_pos);
  526. }
  527. else
  528. {
  529. if (!_IO_in_backup (fp))
  530. _IO_switch_to_wbackup_area (fp);
  531. fp->_wide_data->_IO_read_ptr = fp->_wide_data->_IO_read_end + mark->_pos;
  532. }
  533. return 0;
  534. }
  535. void
  536. _IO_unsave_wmarkers (FILE *fp)
  537. {
  538. struct _IO_marker *mark = fp->_markers;
  539. if (mark)
  540. {
  541. fp->_markers = NULL;
  542. }
  543. if (_IO_have_backup (fp))
  544. _IO_free_wbackup_area (fp);
  545. }