mirror of
https://github.com/llvm-mirror/libcxx.git
synced 2025-10-24 12:02:36 +08:00
git-svn-id: https://llvm.org/svn/llvm-project/libcxx/trunk@178581 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
@@ -807,9 +807,15 @@ basic_filebuf<_CharT, _Traits>::seekoff(off_type __off, ios_base::seekdir __way,
|
|||||||
default:
|
default:
|
||||||
return pos_type(off_type(-1));
|
return pos_type(off_type(-1));
|
||||||
}
|
}
|
||||||
|
#if _WIN32
|
||||||
|
if (fseek(__file_, __width > 0 ? __width * __off : 0, __whence))
|
||||||
|
return pos_type(off_type(-1));
|
||||||
|
pos_type __r = ftell(__file_);
|
||||||
|
#else
|
||||||
if (fseeko(__file_, __width > 0 ? __width * __off : 0, __whence))
|
if (fseeko(__file_, __width > 0 ? __width * __off : 0, __whence))
|
||||||
return pos_type(off_type(-1));
|
return pos_type(off_type(-1));
|
||||||
pos_type __r = ftello(__file_);
|
pos_type __r = ftello(__file_);
|
||||||
|
#endif
|
||||||
__r.state(__st_);
|
__r.state(__st_);
|
||||||
return __r;
|
return __r;
|
||||||
}
|
}
|
||||||
@@ -820,8 +826,13 @@ basic_filebuf<_CharT, _Traits>::seekpos(pos_type __sp, ios_base::openmode)
|
|||||||
{
|
{
|
||||||
if (__file_ == 0 || sync())
|
if (__file_ == 0 || sync())
|
||||||
return pos_type(off_type(-1));
|
return pos_type(off_type(-1));
|
||||||
|
#if _WIN32
|
||||||
|
if (fseek(__file_, __sp, SEEK_SET))
|
||||||
|
return pos_type(off_type(-1));
|
||||||
|
#else
|
||||||
if (fseeko(__file_, __sp, SEEK_SET))
|
if (fseeko(__file_, __sp, SEEK_SET))
|
||||||
return pos_type(off_type(-1));
|
return pos_type(off_type(-1));
|
||||||
|
#endif
|
||||||
__st_ = __sp.state();
|
__st_ = __sp.state();
|
||||||
return __sp;
|
return __sp;
|
||||||
}
|
}
|
||||||
@@ -880,8 +891,13 @@ basic_filebuf<_CharT, _Traits>::sync()
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
#if _WIN32
|
||||||
|
if (fseek(__file_, -__c, SEEK_CUR))
|
||||||
|
return -1;
|
||||||
|
#else
|
||||||
if (fseeko(__file_, -__c, SEEK_CUR))
|
if (fseeko(__file_, -__c, SEEK_CUR))
|
||||||
return -1;
|
return -1;
|
||||||
|
#endif
|
||||||
if (__update_st)
|
if (__update_st)
|
||||||
__st_ = __state;
|
__st_ = __state;
|
||||||
__extbufnext_ = __extbufend_ = __extbuf_;
|
__extbufnext_ = __extbufend_ = __extbuf_;
|
||||||
|
@@ -45,35 +45,108 @@ int vasprintf( char **sptr, const char *__restrict fmt, va_list ap )
|
|||||||
return count;
|
return count;
|
||||||
}
|
}
|
||||||
|
|
||||||
// FIXME: use wcrtomb and avoid copy
|
// Returns >= 0: the number of wide characters found in the multi byte sequence src (of src_size_bytes),
|
||||||
// use mbsrtowcs which is available, first copy first nwc elements of src
|
// that fit in the buffer dst (of max_dest_chars elements size). The count returned excludes the null terminator.
|
||||||
|
// When dst is NULL, no characters are copied and no "out" parameters are updated.
|
||||||
|
// Returns (size_t) -1: an incomplete sequence encountered.
|
||||||
|
// Leaves *src pointing the next character to convert or NULL if a null character was converted from *src.
|
||||||
size_t mbsnrtowcs( wchar_t *__restrict dst, const char **__restrict src,
|
size_t mbsnrtowcs( wchar_t *__restrict dst, const char **__restrict src,
|
||||||
size_t nmc, size_t len, mbstate_t *__restrict ps )
|
size_t src_size_bytes, size_t max_dest_chars, mbstate_t *__restrict ps )
|
||||||
{
|
{
|
||||||
char* local_src = new char[nmc+1];
|
const size_t terminated_sequence = static_cast<size_t>(0);
|
||||||
char* nmcsrc = local_src;
|
//const size_t invalid_sequence = static_cast<size_t>(-1);
|
||||||
strncpy( nmcsrc, *src, nmc );
|
const size_t incomplete_sequence = static_cast< size_t>(-2);
|
||||||
nmcsrc[nmc] = '\0';
|
|
||||||
const size_t result = mbsrtowcs( dst, const_cast<const char **>(&nmcsrc), len, ps );
|
size_t dest_converted = 0;
|
||||||
// propagate error
|
size_t source_converted = 0;
|
||||||
if( nmcsrc == NULL )
|
size_t source_remaining = src_size_bytes;
|
||||||
*src = NULL;
|
size_t result = 0;
|
||||||
delete[] local_src;
|
bool have_result = false;
|
||||||
return result;
|
|
||||||
|
while ( source_remaining ) {
|
||||||
|
if ( dst && dest_converted >= max_dest_chars )
|
||||||
|
break;
|
||||||
|
// Converts one multi byte character.
|
||||||
|
// if result > 0, it's the size in bytes of that character.
|
||||||
|
// othewise if result is zero it indicates the null character has been found.
|
||||||
|
// otherwise it's an error and errno may be set.
|
||||||
|
size_t char_size = mbrtowc( dst ? dst + dest_converted : NULL, *src + source_converted, source_remaining, ps );
|
||||||
|
// Don't do anything to change errno from here on.
|
||||||
|
if ( char_size > 0 ) {
|
||||||
|
source_remaining -= char_size;
|
||||||
|
source_converted += char_size;
|
||||||
|
++dest_converted;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
result = char_size;
|
||||||
|
have_result = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if ( dst ) {
|
||||||
|
if ( have_result && result == terminated_sequence )
|
||||||
|
*src = NULL;
|
||||||
|
else
|
||||||
|
*src += source_converted;
|
||||||
|
}
|
||||||
|
if ( have_result && result != terminated_sequence && result != incomplete_sequence )
|
||||||
|
return static_cast<size_t>(-1);
|
||||||
|
|
||||||
|
return dest_converted;
|
||||||
}
|
}
|
||||||
// FIXME: use wcrtomb and avoid copy
|
|
||||||
// use wcsrtombs which is available, first copy first nwc elements of src
|
// Converts max_source_chars from the wide character buffer pointer to by *src,
|
||||||
|
// into the multi byte character sequence buffer stored at dst which must be dst_size_bytes bytes in size.
|
||||||
|
// Returns >= 0: the number of bytes in the sequence sequence converted frome *src, excluding the null terminator.
|
||||||
|
// Returns size_t(-1) if an error occurs, also sets errno.
|
||||||
|
// If dst is NULL dst_size_bytes is ignored and no bytes are copied to dst and no "out" parameters are updated.
|
||||||
size_t wcsnrtombs( char *__restrict dst, const wchar_t **__restrict src,
|
size_t wcsnrtombs( char *__restrict dst, const wchar_t **__restrict src,
|
||||||
size_t nwc, size_t len, mbstate_t *__restrict ps )
|
size_t max_source_chars, size_t dst_size_bytes, mbstate_t *__restrict ps )
|
||||||
{
|
{
|
||||||
wchar_t* local_src = new wchar_t[nwc];
|
//const size_t invalid_sequence = static_cast<size_t>(-1);
|
||||||
wchar_t* nwcsrc = local_src;
|
|
||||||
wcsncpy(nwcsrc, *src, nwc);
|
size_t source_converted = 0;
|
||||||
nwcsrc[nwc] = '\0';
|
size_t dest_converted = 0;
|
||||||
const size_t result = wcsrtombs( dst, const_cast<const wchar_t **>(&nwcsrc), len, ps );
|
size_t dest_remaining = dst_size_bytes;
|
||||||
// propogate error
|
size_t char_size = 0;
|
||||||
if( nwcsrc == NULL )
|
const errno_t no_error = ( errno_t) 0;
|
||||||
*src = NULL;
|
errno_t result = ( errno_t ) 0;
|
||||||
delete[] nwcsrc;
|
bool have_result = false;
|
||||||
return result;
|
bool terminator_found = false;
|
||||||
|
|
||||||
|
while ( source_converted != max_source_chars ) {
|
||||||
|
if ( ! dest_remaining )
|
||||||
|
break;
|
||||||
|
wchar_t c = (*src)[source_converted];
|
||||||
|
if ( dst )
|
||||||
|
result = wcrtomb_s( &char_size, dst + dest_converted, dest_remaining, c, ps);
|
||||||
|
else
|
||||||
|
result = wcrtomb_s( &char_size, NULL, 0, c, ps);
|
||||||
|
// If result is zero there is no error and char_size contains the size of the multi-byte-sequence converted.
|
||||||
|
// Otherwise result indicates an errno type error.
|
||||||
|
if ( result == no_error ) {
|
||||||
|
if ( c == L'\0' ) {
|
||||||
|
terminator_found = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
++source_converted;
|
||||||
|
if ( dst )
|
||||||
|
dest_remaining -= char_size;
|
||||||
|
dest_converted += char_size;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
have_result = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if ( dst ) {
|
||||||
|
if ( terminator_found )
|
||||||
|
*src = NULL;
|
||||||
|
else
|
||||||
|
*src = *src + source_converted;
|
||||||
|
}
|
||||||
|
if ( have_result && result != no_error ) {
|
||||||
|
errno = result;
|
||||||
|
return static_cast<size_t>(-1);
|
||||||
|
}
|
||||||
|
|
||||||
|
return dest_converted;
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user