VirtualBox

source: vbox/trunk/src/VBox/Runtime/r3/stream.cpp@ 39562

Last change on this file since 39562 was 39395, checked in by vboxsync, 13 years ago

IPRT/stream-r3: Fixed broken char-by-char output in rtStrmWriteLocked().

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 30.8 KB
Line 
1/* $Id: stream.cpp 39395 2011-11-22 13:19:22Z vboxsync $ */
2/** @file
3 * IPRT - I/O Stream.
4 */
5
6/*
7 * Copyright (C) 2006-2007 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27
28
29#if defined(RT_OS_LINUX) /* PORTME: check for the _unlocked functions in stdio.h */
30#define HAVE_FWRITE_UNLOCKED
31#endif
32
33/*******************************************************************************
34* Header Files *
35*******************************************************************************/
36#include <iprt/stream.h>
37#include "internal/iprt.h"
38
39#include <iprt/asm.h>
40#ifndef HAVE_FWRITE_UNLOCKED
41# include <iprt/critsect.h>
42#endif
43#include <iprt/string.h>
44#include <iprt/assert.h>
45#include <iprt/alloc.h>
46#include <iprt/err.h>
47#include <iprt/param.h>
48#include <iprt/string.h>
49
50#include "internal/alignmentchecks.h"
51#include "internal/magics.h"
52
53#include <stdio.h>
54#include <errno.h>
55#ifdef RT_OS_WINDOWS
56# include <io.h>
57# include <fcntl.h>
58# include <Windows.h>
59#endif
60
61
62/*******************************************************************************
63* Structures and Typedefs *
64*******************************************************************************/
65/**
66 * File stream.
67 */
68typedef struct RTSTREAM
69{
70 /** Magic value used to validate the stream. (RTSTREAM_MAGIC) */
71 uint32_t u32Magic;
72 /** File stream error. */
73 int32_t volatile i32Error;
74 /** Pointer to the LIBC file stream. */
75 FILE *pFile;
76 /** Stream is using the current process code set. */
77 bool fCurrentCodeSet;
78 /** Whether the stream was opened in binary mode. */
79 bool fBinary;
80 /** Whether to recheck the stream mode before writing.. */
81 bool fRecheckMode;
82#ifndef HAVE_FWRITE_UNLOCKED
83 /** Critical section for serializing access to the stream. */
84 PRTCRITSECT pCritSect;
85#endif
86} RTSTREAM;
87
88
89/*******************************************************************************
90* Global Variables *
91*******************************************************************************/
92/** The standard input stream. */
93static RTSTREAM g_StdIn =
94{
95 RTSTREAM_MAGIC,
96 0,
97 stdin,
98 /*.fCurrentCodeSet = */ true,
99 /*.fBinary = */ false,
100 /*.fRecheckMode = */ true
101#ifndef HAVE_FWRITE_UNLOCKED
102 , NULL
103#endif
104};
105
106/** The standard error stream. */
107static RTSTREAM g_StdErr =
108{
109 RTSTREAM_MAGIC,
110 0,
111 stderr,
112 /*.fCurrentCodeSet = */ true,
113 /*.fBinary = */ false,
114 /*.fRecheckMode = */ true
115#ifndef HAVE_FWRITE_UNLOCKED
116 , NULL
117#endif
118};
119
120/** The standard output stream. */
121static RTSTREAM g_StdOut =
122{
123 RTSTREAM_MAGIC,
124 0,
125 stdout,
126 /*.fCurrentCodeSet = */ true,
127 /*.fBinary = */ false,
128 /*.fRecheckMode = */ true
129#ifndef HAVE_FWRITE_UNLOCKED
130 , NULL
131#endif
132};
133
134/** Pointer to the standard input stream. */
135RTDATADECL(PRTSTREAM) g_pStdIn = &g_StdIn;
136
137/** Pointer to the standard output stream. */
138RTDATADECL(PRTSTREAM) g_pStdErr = &g_StdErr;
139
140/** Pointer to the standard output stream. */
141RTDATADECL(PRTSTREAM) g_pStdOut = &g_StdOut;
142
143
144#ifndef HAVE_FWRITE_UNLOCKED
145/**
146 * Allocates and acquires the lock for the stream.
147 *
148 * @returns IPRT status.
149 * @param pStream The stream (valid).
150 */
151static int rtStrmAllocLock(PRTSTREAM pStream)
152{
153 Assert(pStream->pCritSect == NULL);
154
155 PRTCRITSECT pCritSect = (PRTCRITSECT)RTMemAlloc(sizeof(*pCritSect));
156 if (!pCritSect)
157 return VERR_NO_MEMORY;
158
159 /* The native stream lock are normally not recursive. */
160 int rc = RTCritSectInitEx(pCritSect, RTCRITSECT_FLAGS_NO_NESTING,
161 NIL_RTLOCKVALCLASS, RTLOCKVAL_SUB_CLASS_NONE, "RTSemSpinMutex");
162 if (RT_SUCCESS(rc))
163 {
164 rc = RTCritSectEnter(pCritSect);
165 if (RT_SUCCESS(rc))
166 {
167 if (RT_LIKELY(ASMAtomicCmpXchgPtr(&pStream->pCritSect, pCritSect, NULL)))
168 return VINF_SUCCESS;
169
170 RTCritSectLeave(pCritSect);
171 }
172 RTCritSectDelete(pCritSect);
173 }
174 RTMemFree(pCritSect);
175
176 /* Handle the lost race case... */
177 pCritSect = ASMAtomicReadPtrT(&pStream->pCritSect, PRTCRITSECT);
178 if (pCritSect)
179 return RTCritSectEnter(pCritSect);
180
181 return rc;
182}
183#endif /* !HAVE_FWRITE_UNLOCKED */
184
185
186/**
187 * Locks the stream. May have to allocate the lock as well.
188 *
189 * @param pStream The stream (valid).
190 */
191DECLINLINE(void) rtStrmLock(PRTSTREAM pStream)
192{
193#ifdef HAVE_FWRITE_UNLOCKED
194 flockfile(pStream->pFile);
195#else
196 if (RT_LIKELY(pStream->pCritSect))
197 RTCritSectEnter(pStream->pCritSect);
198 else
199 rtStrmAllocLock(pStream);
200#endif
201}
202
203
204/**
205 * Unlocks the stream.
206 *
207 * @param pStream The stream (valid).
208 */
209DECLINLINE(void) rtStrmUnlock(PRTSTREAM pStream)
210{
211#ifdef HAVE_FWRITE_UNLOCKED
212 funlockfile(pStream->pFile);
213#else
214 if (RT_LIKELY(pStream->pCritSect))
215 RTCritSectLeave(pStream->pCritSect);
216#endif
217}
218
219
220/**
221 * Opens a file stream.
222 *
223 * @returns iprt status code.
224 * @param pszFilename Path to the file to open.
225 * @param pszMode The open mode. See fopen() standard.
226 * Format: <a|r|w>[+][b|t]
227 * @param ppStream Where to store the opened stream.
228 */
229RTR3DECL(int) RTStrmOpen(const char *pszFilename, const char *pszMode, PRTSTREAM *ppStream)
230{
231 /*
232 * Validate input.
233 */
234 if (!pszMode || !*pszMode)
235 {
236 AssertMsgFailed(("No pszMode!\n"));
237 return VERR_INVALID_PARAMETER;
238 }
239 if (!pszFilename)
240 {
241 AssertMsgFailed(("No pszFilename!\n"));
242 return VERR_INVALID_PARAMETER;
243 }
244 bool fOk = true;
245 bool fBinary = false;
246 switch (*pszMode)
247 {
248 case 'a':
249 case 'w':
250 case 'r':
251 switch (pszMode[1])
252 {
253 case '\0':
254 break;
255
256 case '+':
257 switch (pszMode[2])
258 {
259 case '\0':
260 break;
261
262 //case 't':
263 // break;
264
265 case 'b':
266 fBinary = true;
267 break;
268
269 default:
270 fOk = false;
271 break;
272 }
273 break;
274
275 //case 't':
276 // break;
277
278 case 'b':
279 fBinary = true;
280 break;
281
282 default:
283 fOk = false;
284 break;
285 }
286 break;
287 default:
288 fOk = false;
289 break;
290 }
291 if (!fOk)
292 {
293 AssertMsgFailed(("Invalid pszMode='%s', '<a|r|w>[+][b|t]'\n", pszMode));
294 return VINF_SUCCESS;
295 }
296
297 /*
298 * Allocate the stream handle and try open it.
299 */
300 PRTSTREAM pStream = (PRTSTREAM)RTMemAlloc(sizeof(*pStream));
301 if (pStream)
302 {
303 pStream->u32Magic = RTSTREAM_MAGIC;
304 pStream->i32Error = VINF_SUCCESS;
305 pStream->fCurrentCodeSet = false;
306 pStream->fBinary = fBinary;
307#ifndef HAVE_FWRITE_UNLOCKED
308 pStream->pCritSect = NULL;
309#endif /* HAVE_FWRITE_UNLOCKED */
310 pStream->pFile = fopen(pszFilename, pszMode);
311 if (pStream->pFile)
312 {
313 *ppStream = pStream;
314 return VINF_SUCCESS;
315 }
316 RTMemFree(pStream);
317 return RTErrConvertFromErrno(errno);
318 }
319 return VERR_NO_MEMORY;
320}
321
322
323/**
324 * Opens a file stream.
325 *
326 * @returns iprt status code.
327 * @param pszMode The open mode. See fopen() standard.
328 * Format: <a|r|w>[+][b|t]
329 * @param ppStream Where to store the opened stream.
330 * @param pszFilenameFmt Filename path format string.
331 * @param args Arguments to the format string.
332 */
333RTR3DECL(int) RTStrmOpenFV(const char *pszMode, PRTSTREAM *ppStream, const char *pszFilenameFmt, va_list args)
334{
335 int rc;
336 char szFilename[RTPATH_MAX];
337 size_t cch = RTStrPrintfV(szFilename, sizeof(szFilename), pszFilenameFmt, args);
338 if (cch < sizeof(szFilename))
339 rc = RTStrmOpen(szFilename, pszMode, ppStream);
340 else
341 {
342 AssertMsgFailed(("The filename is too long cch=%d\n", cch));
343 rc = VERR_FILENAME_TOO_LONG;
344 }
345 return rc;
346}
347
348
349/**
350 * Opens a file stream.
351 *
352 * @returns iprt status code.
353 * @param pszMode The open mode. See fopen() standard.
354 * Format: <a|r|w>[+][b|t]
355 * @param ppStream Where to store the opened stream.
356 * @param pszFilenameFmt Filename path format string.
357 * @param ... Arguments to the format string.
358 */
359RTR3DECL(int) RTStrmOpenF(const char *pszMode, PRTSTREAM *ppStream, const char *pszFilenameFmt, ...)
360{
361 va_list args;
362 va_start(args, pszFilenameFmt);
363 int rc = RTStrmOpenFV(pszMode, ppStream, pszFilenameFmt, args);
364 va_end(args);
365 return rc;
366}
367
368
369/**
370 * Closes the specified stream.
371 *
372 * @returns iprt status code.
373 * @param pStream The stream to close.
374 */
375RTR3DECL(int) RTStrmClose(PRTSTREAM pStream)
376{
377 if (!pStream)
378 return VINF_SUCCESS;
379 AssertReturn(RT_VALID_PTR(pStream) && pStream->u32Magic == RTSTREAM_MAGIC, VERR_INVALID_PARAMETER);
380
381 if (!fclose(pStream->pFile))
382 {
383 pStream->u32Magic = 0xdeaddead;
384 pStream->pFile = NULL;
385#ifndef HAVE_FWRITE_UNLOCKED
386 if (pStream->pCritSect)
387 {
388 RTCritSectEnter(pStream->pCritSect);
389 RTCritSectLeave(pStream->pCritSect);
390 RTCritSectDelete(pStream->pCritSect);
391 RTMemFree(pStream->pCritSect);
392 pStream->pCritSect = NULL;
393 }
394#endif
395 RTMemFree(pStream);
396 return VINF_SUCCESS;
397 }
398
399 return RTErrConvertFromErrno(errno);
400}
401
402
403/**
404 * Get the pending error of the stream.
405 *
406 * @returns iprt status code. of the stream.
407 * @param pStream The stream.
408 */
409RTR3DECL(int) RTStrmError(PRTSTREAM pStream)
410{
411 AssertReturn(RT_VALID_PTR(pStream) && pStream->u32Magic == RTSTREAM_MAGIC, VERR_INVALID_PARAMETER);
412 return pStream->i32Error;
413}
414
415
416/**
417 * Clears stream error condition.
418 *
419 * All stream operations save RTStrmClose and this will fail
420 * while an error is asserted on the stream
421 *
422 * @returns iprt status code.
423 * @param pStream The stream.
424 */
425RTR3DECL(int) RTStrmClearError(PRTSTREAM pStream)
426{
427 AssertReturn(RT_VALID_PTR(pStream) && pStream->u32Magic == RTSTREAM_MAGIC, VERR_INVALID_PARAMETER);
428
429 clearerr(pStream->pFile);
430 ASMAtomicWriteS32(&pStream->i32Error, VINF_SUCCESS);
431 return VINF_SUCCESS;
432}
433
434
435RTR3DECL(int) RTStrmSetMode(PRTSTREAM pStream, int fBinary, int fCurrentCodeSet)
436{
437 AssertPtrReturn(pStream, VERR_INVALID_HANDLE);
438 AssertReturn(pStream->u32Magic == RTSTREAM_MAGIC, VERR_INVALID_HANDLE);
439 AssertReturn((unsigned)(fBinary + 1) <= 2, VERR_INVALID_PARAMETER);
440 AssertReturn((unsigned)(fCurrentCodeSet + 1) <= 2, VERR_INVALID_PARAMETER);
441
442 rtStrmLock(pStream);
443
444 if (fBinary != -1)
445 {
446 pStream->fBinary = RT_BOOL(fBinary);
447 pStream->fRecheckMode = true;
448 }
449
450 if (fCurrentCodeSet != -1)
451 pStream->fCurrentCodeSet = RT_BOOL(fCurrentCodeSet);
452
453 rtStrmUnlock(pStream);
454
455 return VINF_SUCCESS;
456}
457
458
459/**
460 * Rewinds the stream.
461 *
462 * Stream errors will be reset on success.
463 *
464 * @returns IPRT status code.
465 *
466 * @param pStream The stream.
467 *
468 * @remarks Not all streams are rewindable and that behavior is currently
469 * undefined for those.
470 */
471RTR3DECL(int) RTStrmRewind(PRTSTREAM pStream)
472{
473 AssertPtrReturn(pStream, VERR_INVALID_HANDLE);
474 AssertReturn(pStream->u32Magic == RTSTREAM_MAGIC, VERR_INVALID_HANDLE);
475
476 int rc;
477 clearerr(pStream->pFile);
478 errno = 0;
479 if (!fseek(pStream->pFile, 0, SEEK_SET))
480 {
481 ASMAtomicWriteS32(&pStream->i32Error, VINF_SUCCESS);
482 rc = VINF_SUCCESS;
483 }
484 else
485 {
486 rc = RTErrConvertFromErrno(errno);
487 ASMAtomicWriteS32(&pStream->i32Error, rc);
488 }
489
490 return rc;
491}
492
493
494/**
495 * Recheck the stream mode.
496 *
497 * @param pStream The stream (locked).
498 */
499static void rtStreamRecheckMode(PRTSTREAM pStream)
500{
501#if defined(RT_OS_WINDOWS) || defined(RT_OS_OS2)
502 int fh = fileno(pStream->pFile);
503 if (fh >= 0)
504 {
505 int fExpected = pStream->fBinary ? _O_BINARY : _O_TEXT;
506 int fActual = _setmode(fh, fExpected);
507 if (fActual != -1 && fExpected != (fActual & (_O_BINARY | _O_TEXT)))
508 {
509 fActual = _setmode(fh, fActual & (_O_BINARY | _O_TEXT));
510 pStream->fBinary = !(fActual & _O_TEXT);
511 }
512 }
513#else
514 NOREF(pStream);
515#endif
516 pStream->fRecheckMode = false;
517}
518
519
520/**
521 * Reads from a file stream.
522 *
523 * @returns iprt status code.
524 * @param pStream The stream.
525 * @param pvBuf Where to put the read bits.
526 * Must be cbRead bytes or more.
527 * @param cbRead Number of bytes to read.
528 * @param pcbRead Where to store the number of bytes actually read.
529 * If NULL cbRead bytes are read or an error is returned.
530 */
531RTR3DECL(int) RTStrmReadEx(PRTSTREAM pStream, void *pvBuf, size_t cbRead, size_t *pcbRead)
532{
533 AssertReturn(RT_VALID_PTR(pStream) && pStream->u32Magic == RTSTREAM_MAGIC, VERR_INVALID_PARAMETER);
534
535 int rc = pStream->i32Error;
536 if (RT_SUCCESS(rc))
537 {
538 if (pStream->fRecheckMode)
539 rtStreamRecheckMode(pStream);
540
541 if (pcbRead)
542 {
543 /*
544 * Can do with a partial read.
545 */
546 *pcbRead = fread(pvBuf, 1, cbRead, pStream->pFile);
547 if ( *pcbRead == cbRead
548 || !ferror(pStream->pFile))
549 return VINF_SUCCESS;
550 if (feof(pStream->pFile))
551 {
552 if (*pcbRead)
553 return VINF_EOF;
554 rc = VERR_EOF;
555 }
556 else if (ferror(pStream->pFile))
557 rc = VERR_READ_ERROR;
558 else
559 {
560 AssertMsgFailed(("This shouldn't happen\n"));
561 rc = VERR_INTERNAL_ERROR;
562 }
563 }
564 else
565 {
566 /*
567 * Must read it all!
568 */
569 if (fread(pvBuf, cbRead, 1, pStream->pFile) == 1)
570 return VINF_SUCCESS;
571
572 /* possible error/eof. */
573 if (feof(pStream->pFile))
574 rc = VERR_EOF;
575 else if (ferror(pStream->pFile))
576 rc = VERR_READ_ERROR;
577 else
578 {
579 AssertMsgFailed(("This shouldn't happen\n"));
580 rc = VERR_INTERNAL_ERROR;
581 }
582 }
583 ASMAtomicWriteS32(&pStream->i32Error, rc);
584 }
585 return rc;
586}
587
588
589/**
590 * Check if the input text is valid UTF-8.
591 *
592 * @returns true/false.
593 * @param pvBuf Pointer to the buffer.
594 * @param cbBuf Size of the buffer.
595 */
596static bool rtStrmIsUtf8Text(const void *pvBuf, size_t cbBuf)
597{
598 NOREF(pvBuf);
599 NOREF(cbBuf);
600 /** @todo not sure this is a good idea... Better redefine RTStrmWrite. */
601 return false;
602}
603
604
605#ifdef RT_OS_WINDOWS
606/**
607 * Check if the stream is for a Window console.
608 *
609 * @returns true / false.
610 * @param pStream The stream.
611 * @param phCon Where to return the console handle.
612 */
613static bool rtStrmIsConsoleUnlocked(PRTSTREAM pStream, HANDLE *phCon)
614{
615 int fh = fileno(pStream->pFile);
616 if (isatty(fh))
617 {
618 DWORD dwMode;
619 HANDLE hCon = (HANDLE)_get_osfhandle(fh);
620 if (GetConsoleMode(hCon, &dwMode))
621 {
622 *phCon = hCon;
623 return true;
624 }
625 }
626 return false;
627}
628#endif /* RT_OS_WINDOWS */
629
630
631/**
632 * Internal write API, stream lock already held.
633 *
634 * @returns IPRT status code.
635 * @param pStream The stream.
636 * @param pvBuf What to write.
637 * @param cbWrite How much to write.
638 * @param pcbWritten Where to optionally return the number of bytes
639 * written.
640 * @param fSureIsText Set if we're sure this is UTF-8 text already.
641 */
642static int rtStrmWriteLocked(PRTSTREAM pStream, const void *pvBuf, size_t cbWrite, size_t *pcbWritten,
643 bool fSureIsText)
644{
645 int rc = pStream->i32Error;
646 if (RT_FAILURE(rc))
647 return rc;
648 if (pStream->fRecheckMode)
649 rtStreamRecheckMode(pStream);
650
651#ifdef RT_OS_WINDOWS
652 /*
653 * Use the unicode console API when possible in order to avoid stuff
654 * getting lost in unnecessary code page translations.
655 */
656 HANDLE hCon;
657 if (rtStrmIsConsoleUnlocked(pStream, &hCon))
658 {
659# ifdef HAVE_FWRITE_UNLOCKED
660 if (!fflush_unlocked(pStream->pFile))
661# else
662 if (!fflush(pStream->pFile))
663# endif
664 {
665 /** @todo Consider buffering later. For now, we'd rather correct output than
666 * fast output. */
667 DWORD cwcWritten = 0;
668 PRTUTF16 pwszSrc = NULL;
669 size_t cwcSrc = 0;
670 rc = RTStrToUtf16Ex((const char *)pvBuf, cbWrite, &pwszSrc, 0, &cwcSrc);
671 if (RT_SUCCESS(rc))
672 {
673 if (!WriteConsoleW(hCon, pwszSrc, (DWORD)cwcSrc, &cwcWritten, NULL))
674 {
675 /* try write char-by-char to avoid heap problem. */
676 cwcWritten = 0;
677 while (cwcWritten != cwcSrc)
678 {
679 DWORD cwcThis;
680 if (!WriteConsoleW(hCon, &pwszSrc[cwcWritten], 1, &cwcThis, NULL))
681 {
682 if (!pcbWritten || cwcWritten == 0)
683 rc = RTErrConvertFromErrno(GetLastError());
684 break;
685 }
686 if (cwcThis != 1) /* Unable to write current char (amount)? */
687 break;
688 cwcWritten++;
689 }
690 }
691 if (RT_SUCCESS(rc))
692 {
693 if (cwcWritten == cwcSrc)
694 {
695 if (pcbWritten)
696 *pcbWritten = cbWrite;
697 }
698 else if (pcbWritten)
699 {
700 PCRTUTF16 pwszCur = pwszSrc;
701 const char *pszCur = (const char *)pvBuf;
702 while ((uintptr_t)(pwszCur - pwszSrc) < cwcWritten)
703 {
704 RTUNICP CpIgnored;
705 RTUtf16GetCpEx(&pwszCur, &CpIgnored);
706 RTStrGetCpEx(&pszCur, &CpIgnored);
707 }
708 *pcbWritten = pszCur - (const char *)pvBuf;
709 }
710 else
711 rc = VERR_WRITE_ERROR;
712 }
713 RTUtf16Free(pwszSrc);
714 }
715 }
716 else
717 rc = RTErrConvertFromErrno(errno);
718 if (RT_FAILURE(rc))
719 ASMAtomicWriteS32(&pStream->i32Error, rc);
720 return rc;
721 }
722#endif /* RT_OS_WINDOWS */
723
724 /*
725 * If we're sure it's text output, convert it from UTF-8 to the current
726 * code page before printing it.
727 *
728 * Note! Partial writes are not supported in this scenario because we
729 * cannot easily report back a written length matching the input.
730 */
731 /** @todo Skip this if the current code set is UTF-8. */
732 if ( pStream->fCurrentCodeSet
733 && !pStream->fBinary
734 && ( fSureIsText
735 || rtStrmIsUtf8Text(pvBuf, cbWrite))
736 )
737 {
738 char *pszSrcFree = NULL;
739 const char *pszSrc = (const char *)pvBuf;
740 if (pszSrc[cbWrite])
741 {
742 pszSrc = pszSrcFree = RTStrDupN(pszSrc, cbWrite);
743 if (pszSrc == NULL)
744 rc = VERR_NO_STR_MEMORY;
745 }
746 if (RT_SUCCESS(rc))
747 {
748 char *pszSrcCurCP;
749 rc = RTStrUtf8ToCurrentCP(&pszSrcCurCP, pszSrc);
750 if (RT_SUCCESS(rc))
751 {
752 size_t cchSrcCurCP = strlen(pszSrcCurCP);
753 IPRT_ALIGNMENT_CHECKS_DISABLE(); /* glibc / mempcpy again */
754#ifdef HAVE_FWRITE_UNLOCKED
755 ssize_t cbWritten = fwrite_unlocked(pszSrcCurCP, cchSrcCurCP, 1, pStream->pFile);
756#else
757 ssize_t cbWritten = fwrite(pszSrcCurCP, cchSrcCurCP, 1, pStream->pFile);
758#endif
759 IPRT_ALIGNMENT_CHECKS_ENABLE();
760 if (cbWritten == 1)
761 {
762 if (pcbWritten)
763 *pcbWritten = cbWrite;
764 }
765#ifdef HAVE_FWRITE_UNLOCKED
766 else if (!ferror_unlocked(pStream->pFile))
767#else
768 else if (!ferror(pStream->pFile))
769#endif
770 {
771 if (pcbWritten)
772 *pcbWritten = 0;
773 }
774 else
775 rc = VERR_WRITE_ERROR;
776 RTStrFree(pszSrcCurCP);
777 }
778 RTStrFree(pszSrcFree);
779 }
780
781 if (RT_FAILURE(rc))
782 ASMAtomicWriteS32(&pStream->i32Error, rc);
783 return rc;
784 }
785
786 /*
787 * Otherwise, just write it as-is.
788 */
789 if (pcbWritten)
790 {
791 IPRT_ALIGNMENT_CHECKS_DISABLE(); /* glibc / mempcpy again */
792#ifdef HAVE_FWRITE_UNLOCKED
793 *pcbWritten = fwrite_unlocked(pvBuf, 1, cbWrite, pStream->pFile);
794#else
795 *pcbWritten = fwrite(pvBuf, 1, cbWrite, pStream->pFile);
796#endif
797 IPRT_ALIGNMENT_CHECKS_ENABLE();
798 if ( *pcbWritten == cbWrite
799#ifdef HAVE_FWRITE_UNLOCKED
800 || !ferror_unlocked(pStream->pFile))
801#else
802 || !ferror(pStream->pFile))
803#endif
804 return VINF_SUCCESS;
805 rc = VERR_WRITE_ERROR;
806 }
807 else
808 {
809 /* Must write it all! */
810 IPRT_ALIGNMENT_CHECKS_DISABLE(); /* glibc / mempcpy again */
811#ifdef HAVE_FWRITE_UNLOCKED
812 size_t cbWritten = fwrite_unlocked(pvBuf, cbWrite, 1, pStream->pFile);
813#else
814 size_t cbWritten = fwrite(pvBuf, cbWrite, 1, pStream->pFile);
815#endif
816 IPRT_ALIGNMENT_CHECKS_ENABLE();
817 if (cbWritten == 1)
818 return VINF_SUCCESS;
819#ifdef HAVE_FWRITE_UNLOCKED
820 if (!ferror_unlocked(pStream->pFile))
821#else
822 if (!ferror(pStream->pFile))
823#endif
824 return VINF_SUCCESS; /* WEIRD! But anyway... */
825
826 rc = VERR_WRITE_ERROR;
827 }
828 ASMAtomicWriteS32(&pStream->i32Error, rc);
829
830 return rc;
831}
832
833
834/**
835 * Internal write API.
836 *
837 * @returns IPRT status code.
838 * @param pStream The stream.
839 * @param pvBuf What to write.
840 * @param cbWrite How much to write.
841 * @param pcbWritten Where to optionally return the number of bytes
842 * written.
843 * @param fSureIsText Set if we're sure this is UTF-8 text already.
844 */
845static int rtStrmWrite(PRTSTREAM pStream, const void *pvBuf, size_t cbWrite, size_t *pcbWritten, bool fSureIsText)
846{
847 rtStrmLock(pStream);
848 int rc = rtStrmWriteLocked(pStream, pvBuf, cbWrite, pcbWritten, fSureIsText);
849 rtStrmUnlock(pStream);
850 return rc;
851}
852
853
854/**
855 * Writes to a file stream.
856 *
857 * @returns iprt status code.
858 * @param pStream The stream.
859 * @param pvBuf Where to get the bits to write from.
860 * @param cbWrite Number of bytes to write.
861 * @param pcbWritten Where to store the number of bytes actually written.
862 * If NULL cbWrite bytes are written or an error is returned.
863 */
864RTR3DECL(int) RTStrmWriteEx(PRTSTREAM pStream, const void *pvBuf, size_t cbWrite, size_t *pcbWritten)
865{
866 AssertReturn(RT_VALID_PTR(pStream) && pStream->u32Magic == RTSTREAM_MAGIC, VERR_INVALID_PARAMETER);
867 return rtStrmWrite(pStream, pvBuf, cbWrite, pcbWritten, false);
868}
869
870
871/**
872 * Reads a character from a file stream.
873 *
874 * @returns The char as an unsigned char cast to int.
875 * @returns -1 on failure.
876 * @param pStream The stream.
877 */
878RTR3DECL(int) RTStrmGetCh(PRTSTREAM pStream)
879{
880 unsigned char ch;
881 int rc = RTStrmReadEx(pStream, &ch, 1, NULL);
882 if (RT_SUCCESS(rc))
883 return ch;
884 return -1;
885}
886
887
888/**
889 * Writes a character to a file stream.
890 *
891 * @returns iprt status code.
892 * @param pStream The stream.
893 * @param ch The char to write.
894 */
895RTR3DECL(int) RTStrmPutCh(PRTSTREAM pStream, int ch)
896{
897 return rtStrmWrite(pStream, &ch, 1, NULL, true /*fSureIsText*/);
898}
899
900
901/**
902 * Writes a string to a file stream.
903 *
904 * @returns iprt status code.
905 * @param pStream The stream.
906 * @param pszString The string to write.
907 * No newlines or anything is appended or prepended.
908 * The terminating '\\0' is not written, of course.
909 */
910RTR3DECL(int) RTStrmPutStr(PRTSTREAM pStream, const char *pszString)
911{
912 size_t cch = strlen(pszString);
913 return rtStrmWrite(pStream, pszString, cch, NULL, true /*fSureIsText*/);
914}
915
916
917/**
918 * Reads a line from a file stream.
919 * A line ends with a '\\n', '\\0' or the end of the file.
920 *
921 * @returns iprt status code.
922 * @returns VINF_BUFFER_OVERFLOW if the buffer wasn't big enough to read an entire line.
923 * @param pStream The stream.
924 * @param pszString Where to store the line.
925 * The line will *NOT* contain any '\\n'.
926 * @param cchString The size of the string buffer.
927 */
928RTR3DECL(int) RTStrmGetLine(PRTSTREAM pStream, char *pszString, size_t cchString)
929{
930 AssertReturn(RT_VALID_PTR(pStream) && pStream->u32Magic == RTSTREAM_MAGIC, VERR_INVALID_PARAMETER);
931 int rc;
932 if (pszString && cchString > 1)
933 {
934 rc = pStream->i32Error;
935 if (RT_SUCCESS(rc))
936 {
937 cchString--; /* save space for the terminator. */
938 rtStrmLock(pStream);
939 for (;;)
940 {
941#ifdef HAVE_FWRITE_UNLOCKED /** @todo darwin + freebsd(?) has fgetc_unlocked but not fwrite_unlocked, optimize... */
942 int ch = fgetc_unlocked(pStream->pFile);
943#else
944 int ch = fgetc(pStream->pFile);
945#endif
946 if (ch == EOF)
947 {
948#ifdef HAVE_FWRITE_UNLOCKED
949 if (feof_unlocked(pStream->pFile))
950#else
951 if (feof(pStream->pFile))
952#endif
953 {
954 rc = VERR_EOF;
955 break;
956 }
957#ifdef HAVE_FWRITE_UNLOCKED
958 if (ferror_unlocked(pStream->pFile))
959#else
960 if (ferror(pStream->pFile))
961#endif
962 rc = VERR_READ_ERROR;
963 else
964 {
965 AssertMsgFailed(("This shouldn't happen\n"));
966 rc = VERR_INTERNAL_ERROR;
967 }
968 break;
969 }
970 if (ch == '\0' || ch == '\n' || ch == '\r')
971 break;
972 *pszString++ = ch;
973 if (--cchString <= 0)
974 {
975 rc = VINF_BUFFER_OVERFLOW;
976 break;
977 }
978 }
979 rtStrmUnlock(pStream);
980
981 *pszString = '\0';
982 if (RT_FAILURE(rc))
983 ASMAtomicWriteS32(&pStream->i32Error, rc);
984 }
985 }
986 else
987 {
988 AssertMsgFailed(("no buffer or too small buffer!\n"));
989 rc = VERR_INVALID_PARAMETER;
990 }
991 return rc;
992}
993
994
995/**
996 * Flushes a stream.
997 *
998 * @returns iprt status code.
999 * @param pStream The stream to flush.
1000 */
1001RTR3DECL(int) RTStrmFlush(PRTSTREAM pStream)
1002{
1003 if (!fflush(pStream->pFile))
1004 return VINF_SUCCESS;
1005 return RTErrConvertFromErrno(errno);
1006}
1007
1008
1009/**
1010 * Output callback.
1011 *
1012 * @returns number of bytes written.
1013 * @param pvArg User argument.
1014 * @param pachChars Pointer to an array of utf-8 characters.
1015 * @param cchChars Number of bytes in the character array pointed to by pachChars.
1016 */
1017static DECLCALLBACK(size_t) rtstrmOutput(void *pvArg, const char *pachChars, size_t cchChars)
1018{
1019 if (cchChars)
1020 rtStrmWriteLocked((PRTSTREAM)pvArg, pachChars, cchChars, NULL, true /*fSureIsText*/);
1021 /* else: ignore termination call. */
1022 return cchChars;
1023}
1024
1025
1026/**
1027 * Prints a formatted string to the specified stream.
1028 *
1029 * @returns Number of bytes printed.
1030 * @param pStream The stream to print to.
1031 * @param pszFormat IPRT format string.
1032 * @param args Arguments specified by pszFormat.
1033 */
1034RTR3DECL(int) RTStrmPrintfV(PRTSTREAM pStream, const char *pszFormat, va_list args)
1035{
1036 AssertReturn(RT_VALID_PTR(pStream) && pStream->u32Magic == RTSTREAM_MAGIC, VERR_INVALID_PARAMETER);
1037 int rc = pStream->i32Error;
1038 if (RT_SUCCESS(rc))
1039 {
1040 rtStrmLock(pStream);
1041// pStream->fShouldFlush = true;
1042 rc = (int)RTStrFormatV(rtstrmOutput, pStream, NULL, NULL, pszFormat, args);
1043 rtStrmUnlock(pStream);
1044 Assert(rc >= 0);
1045 }
1046 else
1047 rc = -1;
1048 return rc;
1049}
1050
1051
1052/**
1053 * Prints a formatted string to the specified stream.
1054 *
1055 * @returns Number of bytes printed.
1056 * @param pStream The stream to print to.
1057 * @param pszFormat IPRT format string.
1058 * @param ... Arguments specified by pszFormat.
1059 */
1060RTR3DECL(int) RTStrmPrintf(PRTSTREAM pStream, const char *pszFormat, ...)
1061{
1062 va_list args;
1063 va_start(args, pszFormat);
1064 int rc = RTStrmPrintfV(pStream, pszFormat, args);
1065 va_end(args);
1066 return rc;
1067}
1068
1069
1070/**
1071 * Prints a formatted string to the standard output stream (g_pStdOut).
1072 *
1073 * @returns Number of bytes printed.
1074 * @param pszFormat IPRT format string.
1075 * @param args Arguments specified by pszFormat.
1076 */
1077RTR3DECL(int) RTPrintfV(const char *pszFormat, va_list args)
1078{
1079 return RTStrmPrintfV(g_pStdOut, pszFormat, args);
1080}
1081
1082
1083/**
1084 * Prints a formatted string to the standard output stream (g_pStdOut).
1085 *
1086 * @returns Number of bytes printed.
1087 * @param pszFormat IPRT format string.
1088 * @param ... Arguments specified by pszFormat.
1089 */
1090RTR3DECL(int) RTPrintf(const char *pszFormat, ...)
1091{
1092 va_list args;
1093 va_start(args, pszFormat);
1094 int rc = RTStrmPrintfV(g_pStdOut, pszFormat, args);
1095 va_end(args);
1096 return rc;
1097}
1098
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette