2003-06-06 21:15:42 +00:00
|
|
|
/*****************************************************************************/
|
|
|
|
/* */
|
|
|
|
/* asserts.c */
|
|
|
|
/* */
|
|
|
|
/* Assertions for the ld65 linker */
|
|
|
|
/* */
|
|
|
|
/* */
|
|
|
|
/* */
|
2011-01-24 22:39:07 +00:00
|
|
|
/* (C) 2003-2011, Ullrich von Bassewitz */
|
2008-03-31 20:54:45 +00:00
|
|
|
/* Roemerstrasse 52 */
|
|
|
|
/* D-70794 Filderstadt */
|
|
|
|
/* EMail: uz@cc65.org */
|
2003-06-06 21:15:42 +00:00
|
|
|
/* */
|
|
|
|
/* */
|
|
|
|
/* This software is provided 'as-is', without any expressed or implied */
|
|
|
|
/* warranty. In no event will the authors be held liable for any damages */
|
|
|
|
/* arising from the use of this software. */
|
|
|
|
/* */
|
|
|
|
/* Permission is granted to anyone to use this software for any purpose, */
|
|
|
|
/* including commercial applications, and to alter it and redistribute it */
|
|
|
|
/* freely, subject to the following restrictions: */
|
|
|
|
/* */
|
|
|
|
/* 1. The origin of this software must not be misrepresented; you must not */
|
|
|
|
/* claim that you wrote the original software. If you use this software */
|
|
|
|
/* in a product, an acknowledgment in the product documentation would be */
|
|
|
|
/* appreciated but is not required. */
|
|
|
|
/* 2. Altered source versions must be plainly marked as such, and must not */
|
|
|
|
/* be misrepresented as being the original software. */
|
|
|
|
/* 3. This notice may not be removed or altered from any source */
|
|
|
|
/* distribution. */
|
|
|
|
/* */
|
|
|
|
/*****************************************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* common */
|
2009-10-04 12:40:19 +00:00
|
|
|
#include "assertion.h"
|
2003-06-06 21:15:42 +00:00
|
|
|
#include "coll.h"
|
|
|
|
#include "xmalloc.h"
|
|
|
|
|
|
|
|
/* ld65 */
|
|
|
|
#include "asserts.h"
|
|
|
|
#include "error.h"
|
2011-01-26 19:42:17 +00:00
|
|
|
#include "expr.h"
|
2003-06-06 21:15:42 +00:00
|
|
|
#include "fileio.h"
|
2011-01-24 22:39:07 +00:00
|
|
|
#include "lineinfo.h"
|
2003-06-06 21:15:42 +00:00
|
|
|
#include "objdata.h"
|
|
|
|
#include "spool.h"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************/
|
2013-05-09 13:56:54 +02:00
|
|
|
/* Data */
|
2003-06-06 21:15:42 +00:00
|
|
|
/*****************************************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
|
2009-10-04 12:40:19 +00:00
|
|
|
/* Assertion struct decl */
|
|
|
|
struct Assertion {
|
2011-01-24 22:39:07 +00:00
|
|
|
Collection LineInfos; /* File position of assertion */
|
2009-10-04 12:40:19 +00:00
|
|
|
ExprNode* Expr; /* Expression to evaluate */
|
|
|
|
AssertAction Action; /* What to do */
|
|
|
|
unsigned Msg; /* Message to print */
|
|
|
|
ObjData* Obj; /* Object file containing the assertion */
|
|
|
|
};
|
|
|
|
|
2003-06-06 21:15:42 +00:00
|
|
|
/* List with all assertions */
|
|
|
|
static Collection Assertions = STATIC_COLLECTION_INITIALIZER;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************/
|
2013-05-09 13:56:54 +02:00
|
|
|
/* Code */
|
2003-06-06 21:15:42 +00:00
|
|
|
/*****************************************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Assertion* ReadAssertion (FILE* F, struct ObjData* O)
|
|
|
|
/* Read an assertion from the given file */
|
|
|
|
{
|
|
|
|
/* Allocate memory */
|
|
|
|
Assertion* A = xmalloc (sizeof (Assertion));
|
|
|
|
|
|
|
|
/* Read the fields from the file */
|
2011-02-10 22:18:24 +00:00
|
|
|
A->LineInfos = EmptyCollection;
|
|
|
|
A->Expr = ReadExpr (F, O);
|
|
|
|
A->Action = (AssertAction) ReadVar (F);
|
|
|
|
A->Msg = MakeGlobalStringId (O, ReadVar (F));
|
2011-01-24 22:39:07 +00:00
|
|
|
ReadLineInfoList (F, O, &A->LineInfos);
|
2003-06-06 21:15:42 +00:00
|
|
|
|
|
|
|
/* Set remaining fields */
|
|
|
|
A->Obj = O;
|
|
|
|
|
|
|
|
/* Add the assertion to the global list */
|
|
|
|
CollAppend (&Assertions, A);
|
|
|
|
|
|
|
|
/* Return the new struct */
|
|
|
|
return A;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void CheckAssertions (void)
|
|
|
|
/* Check all assertions */
|
|
|
|
{
|
|
|
|
unsigned I;
|
|
|
|
|
|
|
|
/* Walk over all assertions */
|
|
|
|
for (I = 0; I < CollCount (&Assertions); ++I) {
|
|
|
|
|
2011-02-10 22:18:24 +00:00
|
|
|
const LineInfo* LI;
|
2011-01-24 22:39:07 +00:00
|
|
|
const char* Module;
|
2011-08-19 10:44:45 +00:00
|
|
|
unsigned Line;
|
2011-01-24 22:39:07 +00:00
|
|
|
|
2003-06-06 21:15:42 +00:00
|
|
|
/* Get the assertion */
|
|
|
|
Assertion* A = CollAtUnchecked (&Assertions, I);
|
|
|
|
|
2009-10-04 12:40:19 +00:00
|
|
|
/* Ignore assertions that shouldn't be handled at link time */
|
|
|
|
if (!AssertAtLinkTime (A->Action)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2011-01-26 19:42:17 +00:00
|
|
|
/* Retrieve the relevant line info for this assertion */
|
|
|
|
LI = CollConstAt (&A->LineInfos, 0);
|
2011-02-10 22:18:24 +00:00
|
|
|
|
2011-01-26 19:42:17 +00:00
|
|
|
/* Get file name and line number from the source */
|
|
|
|
Module = GetSourceName (LI);
|
|
|
|
Line = GetSourceLine (LI);
|
2011-01-24 22:39:07 +00:00
|
|
|
|
2003-06-06 21:15:42 +00:00
|
|
|
/* If the expression is not constant, we're not able to handle it */
|
|
|
|
if (!IsConstExpr (A->Expr)) {
|
2011-08-19 10:44:45 +00:00
|
|
|
Warning ("Cannot evaluate assertion in module `%s', line %u",
|
2011-01-24 22:39:07 +00:00
|
|
|
Module, Line);
|
2003-06-06 21:15:42 +00:00
|
|
|
} else if (GetExprVal (A->Expr) == 0) {
|
|
|
|
|
|
|
|
/* Assertion failed */
|
|
|
|
const char* Message = GetString (A->Msg);
|
|
|
|
|
|
|
|
switch (A->Action) {
|
|
|
|
|
|
|
|
case ASSERT_ACT_WARN:
|
2009-10-04 12:40:19 +00:00
|
|
|
case ASSERT_ACT_LDWARN:
|
2011-08-19 10:44:45 +00:00
|
|
|
Warning ("%s(%u): %s", Module, Line, Message);
|
2003-06-06 21:15:42 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case ASSERT_ACT_ERROR:
|
2009-10-04 12:40:19 +00:00
|
|
|
case ASSERT_ACT_LDERROR:
|
2011-08-19 10:44:45 +00:00
|
|
|
Error ("%s(%u): %s", Module, Line, Message);
|
2003-06-06 21:15:42 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
Internal ("Invalid assertion action (%u) in module `%s', "
|
2011-08-19 10:44:45 +00:00
|
|
|
"line %u (file corrupt?)",
|
2011-01-24 22:39:07 +00:00
|
|
|
A->Action, Module, Line);
|
2003-06-06 21:15:42 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2011-02-10 22:18:24 +00:00
|
|
|
|
2003-06-06 21:15:42 +00:00
|
|
|
|
|
|
|
|