2012-09-19 22:33:13 +00:00
|
|
|
/*
|
|
|
|
* COPYRIGHT: See COPYING in the top level directory
|
|
|
|
* PROJECT: ReactOS CRT
|
|
|
|
* PURPOSE: Implementation of mbstowcs_s
|
2015-09-19 13:50:57 +00:00
|
|
|
* FILE: lib/sdk/crt/string/mbstowcs_s.c
|
2012-09-19 22:33:13 +00:00
|
|
|
* PROGRAMMER: Timo Kreuzer
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <precomp.h>
|
2012-09-27 21:56:13 +00:00
|
|
|
#include <mbstring.h>
|
2012-09-19 22:33:13 +00:00
|
|
|
|
2012-12-16 16:38:36 +00:00
|
|
|
_Success_(return!=EINVAL)
|
2012-09-19 22:33:13 +00:00
|
|
|
_Check_return_opt_
|
2012-12-16 16:38:36 +00:00
|
|
|
_CRTIMP
|
2012-09-19 22:33:13 +00:00
|
|
|
errno_t
|
|
|
|
__cdecl
|
|
|
|
mbstowcs_s(
|
2012-09-27 21:56:13 +00:00
|
|
|
_Out_opt_ size_t *pcchConverted,
|
|
|
|
_Out_writes_to_opt_(sizeInWords, *pcchConverted) wchar_t *pwcstr,
|
2012-09-19 22:33:13 +00:00
|
|
|
_In_ size_t sizeInWords,
|
|
|
|
_In_reads_or_z_(count) const char *pmbstr,
|
|
|
|
_In_ size_t count)
|
|
|
|
{
|
|
|
|
size_t cchMax, cwcWritten;
|
|
|
|
errno_t retval = 0;
|
|
|
|
|
|
|
|
/* Make sure, either we have a target buffer > 0 bytes, or no buffer */
|
|
|
|
if (!MSVCRT_CHECK_PMT( ((sizeInWords != 0) && (pwcstr != 0)) ||
|
|
|
|
((sizeInWords == 0) && (pwcstr == 0)) ))
|
|
|
|
{
|
|
|
|
_set_errno(EINVAL);
|
|
|
|
return EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check if we have a return value pointer */
|
2012-09-27 21:56:13 +00:00
|
|
|
if (pcchConverted)
|
2012-09-19 22:33:13 +00:00
|
|
|
{
|
|
|
|
/* Default to 0 bytes written */
|
2012-09-27 21:56:13 +00:00
|
|
|
*pcchConverted = 0;
|
2012-09-19 22:33:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!MSVCRT_CHECK_PMT((count == 0) || (pmbstr != 0)))
|
|
|
|
{
|
|
|
|
_set_errno(EINVAL);
|
|
|
|
return EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check if there is anything to do */
|
|
|
|
if ((pwcstr == 0) && (pmbstr == 0))
|
|
|
|
{
|
|
|
|
_set_errno(EINVAL);
|
|
|
|
return EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check if we have a multibyte string */
|
|
|
|
if (pmbstr)
|
|
|
|
{
|
|
|
|
/* Check if we also have a wchar buffer */
|
|
|
|
if (pwcstr)
|
|
|
|
{
|
|
|
|
/* Calculate the maximum the we can write */
|
|
|
|
cchMax = (count < sizeInWords) ? count + 1 : sizeInWords;
|
|
|
|
|
|
|
|
/* Now do the conversion */
|
|
|
|
cwcWritten = mbstowcs(pwcstr, pmbstr, cchMax);
|
|
|
|
|
|
|
|
/* Check if the buffer was not zero terminated */
|
|
|
|
if (cwcWritten == cchMax)
|
|
|
|
{
|
|
|
|
/* Check if we reached the max size of the dest buffer */
|
|
|
|
if (cwcWritten == sizeInWords)
|
|
|
|
{
|
|
|
|
/* Does the caller allow this? */
|
|
|
|
if (count != _TRUNCATE)
|
|
|
|
{
|
|
|
|
/* Not allowed, truncate to 0 length */
|
|
|
|
pwcstr[0] = L'\0';
|
|
|
|
|
|
|
|
/* Return error */
|
|
|
|
_set_errno(ERANGE);
|
|
|
|
return ERANGE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Inform the caller about truncation */
|
|
|
|
retval = STRUNCATE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* zero teminate the buffer */
|
|
|
|
pwcstr[cwcWritten - 1] = L'\0';
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* The buffer is zero terminated, count the terminating char */
|
|
|
|
cwcWritten++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Get the length of the string, plus 0 terminator */
|
2012-09-27 22:01:26 +00:00
|
|
|
cwcWritten = _mbsnlen((const unsigned char *)pmbstr, count) + 1;
|
2012-09-19 22:33:13 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
cwcWritten = count + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check if we have a return value pointer */
|
2012-09-27 21:56:13 +00:00
|
|
|
if (pcchConverted)
|
2012-09-19 22:33:13 +00:00
|
|
|
{
|
|
|
|
/* Default to 0 bytes written */
|
2012-09-27 21:56:13 +00:00
|
|
|
*pcchConverted = cwcWritten;
|
2012-09-19 22:33:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return retval;
|
|
|
|
}
|