8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-27 20:03:03 +01:00
firebird-mirror/src/common/DynamicStrings.cpp
alexpeshkoff 139331c7e3 Enhanced memory leaks search - changed FB_NEW to FB_NEW_POOL,
plain new to FB_NEW, making allocated with it blocks tagged with file/line.
Make related code slightly better readable.
2015-10-12 14:26:00 +00:00

146 lines
3.1 KiB
C++

/*
* PROGRAM: Firebird status vector support.
* MODULE: DynamicStrings.cpp
* DESCRIPTION: Dynamically store strings in status vector.
*
* The contents of this file are subject to the Initial
* Developer's Public License Version 1.0 (the "License");
* you may not use this file except in compliance with the
* License. You may obtain a copy of the License at
* http://www.ibphoenix.com/main.nfs?a=ibphoenix&page=ibp_idpl.
*
* Software distributed under the License is distributed AS IS,
* WITHOUT WARRANTY OF ANY KIND, either express or implied.
* See the License for the specific language governing rights
* and limitations under the License.
*
* The Original Code was created by Alex Peshkov
* for the Firebird Open Source RDBMS project.
*
* Copyright (c) 2015 Alex Peshkov <peshkoff at mail dot ru>
* and all contributors signed below.
*
* All Rights Reserved.
* Contributor(s): ______________________________________.
*
*
*/
#include "firebird.h"
#include "DynamicStrings.h"
#include "utils_proto.h"
#include "gen/iberror.h"
#include "classes/alloc.h"
#include <string.h>
namespace Firebird {
unsigned makeDynamicStrings(unsigned length, ISC_STATUS* const dst, const ISC_STATUS* const src) throw(BadAlloc)
{
const ISC_STATUS* end = &src[length];
// allocate space for strings
size_t len = 0;
for (const ISC_STATUS* from = src; from < end; ++from)
{
const ISC_STATUS type = *from++;
if (from == end || type == isc_arg_end)
{
end = from - 1;
break;
}
switch (type)
{
case isc_arg_cstring:
if (from + 1 >= end)
{
end = from - 1;
break;
}
len += *from++;
len++;
break;
case isc_arg_string:
case isc_arg_interpreted:
case isc_arg_sql_state:
len += strlen(reinterpret_cast<const char*>(*from));
len++;
break;
}
}
char* string = len ? FB_NEW_POOL(*getDefaultMemoryPool()) char[len] : NULL;
ISC_STATUS* to = dst;
// copy status vector saving strings in local buffer
for (const ISC_STATUS* from = src; from < end; ++from)
{
const ISC_STATUS type = *from++;
*to++ = type == isc_arg_cstring ? isc_arg_string : type;
switch (type)
{
case isc_arg_cstring:
fb_assert(string);
*to++ = (ISC_STATUS)(IPTR) string;
memcpy(string, reinterpret_cast<const char*>(from[1]), from[0]);
string += *from++;
*string++ = '\0';
break;
case isc_arg_string:
case isc_arg_interpreted:
case isc_arg_sql_state:
fb_assert(string);
*to++ = (ISC_STATUS)(IPTR) string;
strcpy(string, reinterpret_cast<const char*>(*from));
string += strlen(string);
string++;
break;
default:
*to++ = *from;
break;
}
}
*to++ = isc_arg_end;
return (to - dst) - 1;
}
char* findDynamicStrings(unsigned length, ISC_STATUS* ptr) throw()
{
while (length--)
{
const ISC_STATUS type = *ptr++;
if (type == isc_arg_end)
break;
switch (type)
{
case isc_arg_cstring:
fb_assert(false); // CVC: according to the new logic, this case cannot happen
ptr++;
case isc_arg_string:
case isc_arg_interpreted:
case isc_arg_sql_state:
return reinterpret_cast<char*>(*ptr);
default:
ptr++;
break;
}
}
return NULL;
}
} // namespace Firebird