2006-10-30 22:04:34 +01:00
|
|
|
/*
|
|
|
|
* 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 Vlad Horsun
|
|
|
|
* for the Firebird Open Source RDBMS project.
|
|
|
|
*
|
|
|
|
* Copyright (c) 2006 Vlad Horsun <hvlad@users.sourceforge.net>
|
|
|
|
* and all contributors signed below.
|
|
|
|
*
|
|
|
|
* All Rights Reserved.
|
|
|
|
* Contributor(s): ______________________________________.
|
|
|
|
*/
|
|
|
|
|
2006-11-05 05:01:30 +01:00
|
|
|
#include "firebird.h"
|
2010-04-02 23:48:15 +02:00
|
|
|
#include "../jrd/Attachment.h"
|
2006-10-30 22:04:34 +01:00
|
|
|
#include "../jrd/DebugInterface.h"
|
|
|
|
#include "../jrd/blb_proto.h"
|
|
|
|
|
|
|
|
using namespace Jrd;
|
|
|
|
using namespace Firebird;
|
|
|
|
|
2011-10-03 00:11:41 +02:00
|
|
|
void DBG_parse_debug_info(thread_db* tdbb, bid* blob_id, DbgInfo& dbgInfo)
|
2006-10-30 22:04:34 +01:00
|
|
|
{
|
2010-04-02 23:48:15 +02:00
|
|
|
Jrd::Attachment* attachment = tdbb->getAttachment();
|
|
|
|
|
2012-02-15 04:34:21 +01:00
|
|
|
blb* blob = blb::open(tdbb, attachment->getSysTransaction(), blob_id);
|
2009-08-05 14:31:54 +02:00
|
|
|
const ULONG length = blob->blb_length;
|
2011-01-29 18:24:29 +01:00
|
|
|
HalfStaticArray<UCHAR, 128> tmp;
|
2006-10-30 22:04:34 +01:00
|
|
|
|
|
|
|
UCHAR* temp = tmp.getBuffer(length);
|
2012-02-07 04:17:52 +01:00
|
|
|
blob->BLB_get_data(tdbb, temp, length);
|
2006-10-30 22:04:34 +01:00
|
|
|
|
2007-01-20 15:18:18 +01:00
|
|
|
DBG_parse_debug_info(length, temp, dbgInfo);
|
|
|
|
}
|
|
|
|
|
2011-10-03 00:11:41 +02:00
|
|
|
void DBG_parse_debug_info(ULONG length, const UCHAR* data, DbgInfo& dbgInfo)
|
2007-01-20 15:18:18 +01:00
|
|
|
{
|
2008-03-06 10:36:45 +01:00
|
|
|
const UCHAR* const end = data + length;
|
2006-10-30 22:04:34 +01:00
|
|
|
bool bad_format = false;
|
|
|
|
|
2013-09-05 09:38:56 +02:00
|
|
|
if ((*data++ != fb_dbg_version) || (end[-1] != fb_dbg_end))
|
2006-10-30 22:04:34 +01:00
|
|
|
bad_format = true;
|
2013-09-05 09:38:56 +02:00
|
|
|
|
|
|
|
UCHAR version = UCHAR(0);
|
|
|
|
|
|
|
|
if (!bad_format)
|
|
|
|
{
|
|
|
|
version = *data++;
|
|
|
|
|
|
|
|
if (!version || version > CURRENT_DBG_INFO_VERSION)
|
|
|
|
bad_format = true;
|
2006-10-30 22:04:34 +01:00
|
|
|
}
|
|
|
|
|
2008-12-05 02:20:14 +01:00
|
|
|
while (!bad_format && (data < end))
|
2006-10-30 22:04:34 +01:00
|
|
|
{
|
2011-10-16 22:36:07 +02:00
|
|
|
UCHAR code = *data++;
|
|
|
|
|
|
|
|
switch (code)
|
2006-10-30 22:04:34 +01:00
|
|
|
{
|
|
|
|
case fb_dbg_map_src2blr:
|
2008-01-16 09:31:31 +01:00
|
|
|
{
|
2013-09-05 09:38:56 +02:00
|
|
|
const unsigned length =
|
|
|
|
(version == DBG_INFO_VERSION_1) ? 6 : 12;
|
|
|
|
|
|
|
|
if (data + length > end)
|
2011-01-29 18:24:29 +01:00
|
|
|
{
|
2008-01-16 09:31:31 +01:00
|
|
|
bad_format = true;
|
|
|
|
break;
|
|
|
|
}
|
2006-10-30 22:04:34 +01:00
|
|
|
|
2008-01-16 09:31:31 +01:00
|
|
|
MapBlrToSrcItem i;
|
2013-09-05 09:38:56 +02:00
|
|
|
|
2008-01-16 09:31:31 +01:00
|
|
|
i.mbs_src_line = *data++;
|
|
|
|
i.mbs_src_line |= *data++ << 8;
|
2013-09-05 09:38:56 +02:00
|
|
|
|
|
|
|
if (version > DBG_INFO_VERSION_1)
|
|
|
|
{
|
|
|
|
i.mbs_src_line |= *data++ << 16;
|
|
|
|
i.mbs_src_line |= *data++ << 24;
|
|
|
|
}
|
2006-10-30 22:04:34 +01:00
|
|
|
|
2008-01-16 09:31:31 +01:00
|
|
|
i.mbs_src_col = *data++;
|
|
|
|
i.mbs_src_col |= *data++ << 8;
|
2013-09-05 09:38:56 +02:00
|
|
|
|
|
|
|
if (version > DBG_INFO_VERSION_1)
|
|
|
|
{
|
|
|
|
i.mbs_src_col |= *data++ << 16;
|
|
|
|
i.mbs_src_col |= *data++ << 24;
|
|
|
|
}
|
2006-10-30 22:04:34 +01:00
|
|
|
|
2008-01-16 09:31:31 +01:00
|
|
|
i.mbs_offset = *data++;
|
|
|
|
i.mbs_offset |= *data++ << 8;
|
2013-09-05 09:38:56 +02:00
|
|
|
|
|
|
|
if (version > DBG_INFO_VERSION_1)
|
|
|
|
{
|
|
|
|
i.mbs_offset |= *data++ << 16;
|
|
|
|
i.mbs_offset |= *data++ << 24;
|
|
|
|
}
|
2006-10-30 22:04:34 +01:00
|
|
|
|
2008-01-16 09:31:31 +01:00
|
|
|
dbgInfo.blrToSrc.add(i);
|
|
|
|
}
|
|
|
|
break;
|
2006-10-30 22:04:34 +01:00
|
|
|
|
|
|
|
case fb_dbg_map_varname:
|
2014-05-19 21:06:47 +02:00
|
|
|
case fb_dbg_map_curname:
|
2008-01-16 09:31:31 +01:00
|
|
|
{
|
2011-01-29 18:24:29 +01:00
|
|
|
if (data + 3 > end)
|
|
|
|
{
|
2008-01-16 09:31:31 +01:00
|
|
|
bad_format = true;
|
|
|
|
break;
|
|
|
|
}
|
2006-10-30 22:04:34 +01:00
|
|
|
|
2014-05-19 21:06:47 +02:00
|
|
|
// variable/cursor number
|
2008-01-16 09:31:31 +01:00
|
|
|
USHORT index = *data++;
|
2010-06-17 17:32:11 +02:00
|
|
|
index |= *data++ << 8;
|
2006-10-30 22:04:34 +01:00
|
|
|
|
2014-05-19 21:06:47 +02:00
|
|
|
// variable/cursor name string length
|
2008-01-16 09:31:31 +01:00
|
|
|
USHORT length = *data++;
|
2006-10-30 22:04:34 +01:00
|
|
|
|
2011-01-29 18:24:29 +01:00
|
|
|
if (data + length > end)
|
|
|
|
{
|
2008-01-16 09:31:31 +01:00
|
|
|
bad_format = true;
|
|
|
|
break;
|
|
|
|
}
|
2007-01-17 02:19:01 +01:00
|
|
|
|
2014-05-19 21:06:47 +02:00
|
|
|
if (code == fb_dbg_map_varname)
|
|
|
|
dbgInfo.varIndexToName.put(index, MetaName((const TEXT*) data, length));
|
|
|
|
else
|
|
|
|
dbgInfo.curIndexToName.put(index, MetaName((const TEXT*) data, length));
|
2007-01-17 02:19:01 +01:00
|
|
|
|
2014-05-19 21:06:47 +02:00
|
|
|
// variable/cursor name string
|
2008-01-16 09:31:31 +01:00
|
|
|
data += length;
|
|
|
|
}
|
|
|
|
break;
|
2007-01-17 02:19:01 +01:00
|
|
|
|
|
|
|
case fb_dbg_map_argument:
|
2008-01-16 09:31:31 +01:00
|
|
|
{
|
2011-01-29 18:24:29 +01:00
|
|
|
if (data + 4 > end)
|
|
|
|
{
|
2008-01-16 09:31:31 +01:00
|
|
|
bad_format = true;
|
|
|
|
break;
|
|
|
|
}
|
2007-01-17 02:19:01 +01:00
|
|
|
|
2008-01-16 09:31:31 +01:00
|
|
|
ArgumentInfo info;
|
2007-01-17 02:19:01 +01:00
|
|
|
|
2008-01-16 09:31:31 +01:00
|
|
|
// argument type
|
|
|
|
info.type = *data++;
|
2007-01-17 02:19:01 +01:00
|
|
|
|
2008-01-16 09:31:31 +01:00
|
|
|
// argument number
|
|
|
|
info.index = *data++;
|
2010-06-17 17:32:11 +02:00
|
|
|
info.index |= *data++ << 8;
|
2007-01-17 02:19:01 +01:00
|
|
|
|
2008-01-16 09:31:31 +01:00
|
|
|
// argument name string length
|
|
|
|
USHORT length = *data++;
|
2007-01-17 02:19:01 +01:00
|
|
|
|
2011-01-29 18:24:29 +01:00
|
|
|
if (data + length > end)
|
|
|
|
{
|
2008-01-16 09:31:31 +01:00
|
|
|
bad_format = true;
|
|
|
|
break;
|
|
|
|
}
|
2007-01-17 02:19:01 +01:00
|
|
|
|
2008-01-16 09:31:31 +01:00
|
|
|
dbgInfo.argInfoToName.put(info, MetaName((const TEXT*) data, length));
|
2007-01-17 02:19:01 +01:00
|
|
|
|
2008-01-16 09:31:31 +01:00
|
|
|
// variable name string
|
|
|
|
data += length;
|
|
|
|
}
|
|
|
|
break;
|
2006-10-30 22:04:34 +01:00
|
|
|
|
2011-10-03 00:11:41 +02:00
|
|
|
case fb_dbg_subproc:
|
2011-10-16 22:36:07 +02:00
|
|
|
case fb_dbg_subfunc:
|
2011-10-03 00:11:41 +02:00
|
|
|
{
|
2013-09-05 09:38:56 +02:00
|
|
|
if (version == DBG_INFO_VERSION_1 || data >= end)
|
2011-10-03 00:11:41 +02:00
|
|
|
{
|
|
|
|
bad_format = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// argument name string length
|
|
|
|
ULONG length = *data++;
|
|
|
|
|
|
|
|
if (data + length >= end)
|
|
|
|
{
|
|
|
|
bad_format = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
MetaName name((const TEXT*) data, length);
|
|
|
|
data += length;
|
|
|
|
|
2013-09-05 09:38:56 +02:00
|
|
|
if (data + 4 >= end)
|
2011-10-03 00:11:41 +02:00
|
|
|
{
|
|
|
|
bad_format = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
length = *data++;
|
|
|
|
length |= *data++ << 8;
|
|
|
|
length |= *data++ << 16;
|
|
|
|
length |= *data++ << 24;
|
|
|
|
|
|
|
|
if (data + length >= end)
|
|
|
|
{
|
|
|
|
bad_format = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2015-10-12 16:26:00 +02:00
|
|
|
AutoPtr<DbgInfo> sub(FB_NEW_POOL(dbgInfo.getPool()) DbgInfo(dbgInfo.getPool()));
|
2011-10-03 00:11:41 +02:00
|
|
|
DBG_parse_debug_info(length, data, *sub);
|
|
|
|
data += length;
|
|
|
|
|
2011-10-16 22:36:07 +02:00
|
|
|
if (code == fb_dbg_subproc)
|
|
|
|
dbgInfo.subProcs.put(name, sub.release());
|
|
|
|
else
|
|
|
|
dbgInfo.subFuncs.put(name, sub.release());
|
2011-10-03 00:11:41 +02:00
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2006-10-30 22:04:34 +01:00
|
|
|
case fb_dbg_end:
|
2007-01-20 15:18:18 +01:00
|
|
|
if (data != end)
|
2006-10-30 22:04:34 +01:00
|
|
|
bad_format = true;
|
2008-01-16 09:31:31 +01:00
|
|
|
break;
|
2006-10-30 22:04:34 +01:00
|
|
|
|
|
|
|
default:
|
|
|
|
bad_format = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-03-06 10:36:45 +01:00
|
|
|
if (bad_format || data != end)
|
2006-10-30 22:04:34 +01:00
|
|
|
{
|
2007-01-20 15:18:18 +01:00
|
|
|
dbgInfo.clear();
|
2008-08-27 14:20:47 +02:00
|
|
|
ERR_post_warning(Arg::Warning(isc_bad_debug_format));
|
2006-10-30 22:04:34 +01:00
|
|
|
}
|
|
|
|
}
|