gecko-dev/lib/libparse/pa_parse.c

2017 lines
49 KiB
C

/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*-
*
* The contents of this file are subject to the Netscape Public License
* Version 1.0 (the "NPL"); you may not use this file except in
* compliance with the NPL. You may obtain a copy of the NPL at
* http://www.mozilla.org/NPL/
*
* Software distributed under the NPL is distributed on an "AS IS" basis,
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the NPL
* for the specific language governing rights and limitations under the
* NPL.
*
* The Initial Developer of this code under the NPL is Netscape
* Communications Corporation. Portions created by Netscape are
* Copyright (C) 1998 Netscape Communications Corporation. All Rights
* Reserved.
*/
#include "pa_parse.h"
#include <stdio.h>
#include "merrors.h"
#include "net.h"
#include "hk_funcs.h"
#include "libevent.h"
#include "intl_csi.h"
extern int MK_OUT_OF_MEMORY;
#ifdef PROFILE
#pragma profile on
#endif
#ifdef XP_WIN16
#define HOLD_BUF_UNIT 32000
#define SIZE_LIMIT 32000
#else
#define HOLD_BUF_UNIT 16384
#endif /* XP_WIN16 */
#define WRITE_READY_SIZE (unsigned int) 8192
/*
* Function to call with parsed tag elements.
* It should be initialized by a call to PA_ParserInit*().
*/
static intn (*PA_ParsedTag)(void *data_object, PA_Tag *tags, intn status) = NULL;
typedef struct pa_DocDataList_struct {
MWContext* window_id;
pa_DocData *doc_data;
struct pa_DocDataList_struct *next;
} pa_DocDataList;
static pa_DocDataList *DocDataList = NULL;
static pa_DocData *pa_FetchDocData(MWContext *window_id);
static Bool pa_RemoveDocData(pa_DocData *target_doc_data);
static Bool pa_StoreDocData(MWContext *window_id, pa_DocData *new_doc_data);
/*************************
* The following is to speed up case conversion
* to allow faster checking of caseless equal among strings.
*************************/
#ifndef NON_ASCII_STRINGS
unsigned char lower_lookup[256]={
0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20,21,22,23,24,25,26,
27,28,29,30,31,32,33,34,35,36,37,38,39,40,41,42,43,44,45,46,47,48,49,50,
51,52,53,54,55,56,57,58,59,60,61,62,63,64,
97,98,99,100,101,102,103,104,105,106,107,108,109,
110,111,112,113,114,115,116,117,118,119,120,121,122,
91,92,93,94,95,96,97,98,99,100,101,102,103,104,105,106,107,108,109,110,
111,112,113,114,115,116,117,118,119,120,121,122,123,124,125,126,127,128,
129,130,131,132,133,134,135,136,137,138,139,140,141,142,143,144,145,146,
147,148,149,150,151,152,153,154,155,156,157,158,159,160,161,162,163,164,
165,166,167,168,169,170,171,172,173,174,175,176,177,178,179,180,181,182,
183,184,185,186,187,188,189,190,191,192,193,194,195,196,197,198,199,200,
201,202,203,204,205,206,207,208,209,210,211,212,213,214,215,216,217,218,
219,220,221,222,223,224,225,226,227,228,229,230,231,232,233,234,235,236,
237,238,239,240,241,242,243,244,245,246,247,248,249,250,251,252,253,254,
255};
#endif /* not NON_ASCII_STRINGS */
/*************************************
* Function: pa_caseless_equal
*
* Description: This function will compare two
* strings, similar to strcmp(), but ignoring the
* case of the letters A-Z.
*
* Params: Takes two \0 terminated strings.
*
* Returns: 1 if strings are equal, 0 if not.
*************************************/
#if 0
static intn
pa_caseless_equal(char *string_1, char *string_2)
{
/*
* If either is NULL, they are not equal, even if both are NULL
*/
if ((string_1 == NULL)||(string_2 == NULL))
{
return(0);
}
/*
* While not at the end of the string, if they ever differ
* they are not equal.
*/
while ((*string_1 != '\0')&&(*string_2 != '\0'))
{
if (TOLOWER((unsigned char) *string_1) != TOLOWER((unsigned char) *string_2))
{
return(0);
}
string_1++;
string_2++;
}
/*
* One of the strings has ended, if they are both ended, then they
* are equal, otherwise not.
*/
if ((*string_1 == '\0')&&(*string_2 == '\0'))
{
return(1);
}
else
{
return(0);
}
}
#endif
/*************************************
* Function: pa_TagEqual
*
* Description: This function is a special purpose caseless compare
* to save me a few cycles of performance.
* Since we know the first string is a predefined TAG
* we are guaranteeing it will always be in lower case,
* thus we don't need to TOLOWER its characters as we
* compare them.
*
* Params: Takes two \0 terminated strings. The first, being a predefined TAG
* is guaranteed to be all in lower case.
*
* Returns: 1 if strings are equal, 0 if not.
*************************************/
intn
pa_TagEqual(char *tag, char *str)
{
/*
* If str is NULL, they are not equal, tag cannot be NULL.
*/
if (str == NULL)
{
return(0);
}
/*
* While not at the end of the string, if they ever differ
* they are not equal.
*/
while ((*tag != '\0')&&(*str != '\0'))
{
if ((int)(*tag) != TOLOWER((unsigned char) *str))
{
return(0);
}
tag++;
str++;
}
/*
* One of the strings has ended, if they are both ended, then they
* are equal, otherwise not.
*/
if ((*tag == '\0')&&(*str == '\0'))
{
return(1);
}
else
{
return(0);
}
}
/*************************************
* Function: PA_FreeTag
*
* Description: This function frees up all memory associated
* with a PA_Tag structure, including the structure
* itself.
*
* Params: Takes pointer to a PA_Tag structure.
*
* Returns: none.
*************************************/
void
PA_FreeTag(PA_Tag *tag)
{
/*
* Nothing to do for already freed tags.
*/
if (tag == NULL)
{
return;
}
/*
* If we have data, free it.
*/
if (tag->data != NULL)
{
PA_FREE(tag->data);
}
/*
* Free the tag structure.
*/
XP_DELETE(tag);
}
static int32 doc_id_gen = 0; /* generator for document identifiers */
/*************************************
* Function: pa_new_document
*
* Description: register a new document, create and initialize
* the pa_DocData structure for it.
*
* Params: Takes a unique document id, and the URL_Struct for this doc.
*
* Returns: a pointer to the new pa_DocData structure, already
* initialized, and with the doc_id filled in.
* Return NULL on failure.
*************************************/
static pa_DocData *
pa_new_document(FO_Present_Types format_out,
MWContext *window_id,
PA_OutputFunction *output_func,
URL_Struct *url_struct)
{
pa_DocData *doc_data, *old_doc_data;
Bool is_inline_stream;
if (format_out == FO_PRESENT_INLINE) {
is_inline_stream = TRUE;
old_doc_data = pa_FetchDocData(window_id);
}
else {
/* Added by Lou:
* This will interrupt anything else trying to go into
* this same window, so that we have a clear path
* to load this new document
*/
NET_SilentInterruptWindow(window_id);
is_inline_stream = FALSE;
}
doc_data = XP_NEW(pa_DocData);
if (doc_data == NULL)
{
return(NULL);
}
/*
* Allocate a static hold buffer. This will
* save on malloc calls in the long run.
*/
doc_data->hold_buf = XP_ALLOC_BLOCK(HOLD_BUF_UNIT * sizeof(char));
if (doc_data->hold_buf == NULL)
{
XP_DELETE(doc_data);
return(NULL);
}
/*
* Now that we can't fail, create the unique document ID.
*/
if (is_inline_stream) {
/*
* If we're still parsing the original HTML stream, then
* just get some of the required information from the
* existing doc_data. Otherwise, we need to get it from
* MWContext - is there a better way to do this?
*/
if (old_doc_data) {
doc_data->doc_id = old_doc_data->doc_id;
doc_data->layout_state = old_doc_data->layout_state;
}
else {
doc_data->doc_id = window_id->doc_id;
doc_data->layout_state = NULL;
}
}
else {
doc_data->doc_id = ++doc_id_gen;
doc_data->layout_state = NULL;
}
doc_data->window_id = window_id;
doc_data->output_tag = output_func;
doc_data->hold = 0;
doc_data->hold_size = HOLD_BUF_UNIT;
doc_data->hold_len = 0;
doc_data->brute_tag = P_UNKNOWN;
doc_data->comment_bytes = 0;
doc_data->lose_newline = FALSE;
if (url_struct->address == NULL)
{
doc_data->url = NULL;
}
else
{
doc_data->url = XP_STRDUP(url_struct->address);
}
if ((url_struct->cache_file != NULL)||(url_struct->memory_copy != NULL))
{
doc_data->from_net = FALSE;
}
else
{
doc_data->from_net = TRUE;
}
/*
* A NET_SUPER_RELOAD should always make everything reload, so no
* matter what, act as if it all came new from the net.
*/
if (url_struct->force_reload == NET_SUPER_RELOAD)
{
doc_data->from_net = TRUE;
}
/* save the url struct and make sure it doesn't disappear on us */
doc_data->url_struct = url_struct;
NET_HoldURLStruct(doc_data->url_struct);
doc_data->edit_buffer = NULL;
doc_data->is_inline_stream = is_inline_stream;
if (!doc_data->is_inline_stream || !old_doc_data)
pa_StoreDocData(window_id, doc_data);
doc_data->overflow_stack = 0;
doc_data->overflow_depth = 0;
doc_data->stream_status = 0;
doc_data->stream_count = 0;
return(doc_data);
}
/*
*
*/
unsigned int
pa_ParseWriteReady (NET_StreamClass *stream)
{
pa_DocData *doc_data = (pa_DocData *) stream->data_object;
#if !defined(XP_UNIX) && !defined(XP_WIN32)
if (doc_data->overflow_depth)
return 0;
#endif
return WRITE_READY_SIZE;
}
static pa_DocData *
pa_check_doc_data_count(NET_StreamClass *stream)
{
pa_DocData *doc_data=stream->data_object;
/* netlib will no longer send us data */
if (doc_data->stream_count + doc_data->overflow_depth <= 0) {
if (doc_data->stream_status != 0) {
PA_MDLAbort(stream, doc_data->stream_status);
}
else {
PA_MDLComplete(stream);
}
return NULL;
}
return doc_data;
}
/*
* Flush the data out of the current overflow buffer and send it up through
* the parser. After we push the data through PA_ParseBlock()
* the overflow buffer is empty, any unparsed data will be left
* in the hold buffer
*/
void
pa_FlushOverflow(NET_StreamClass *stream)
{
pa_DocData *doc_data=stream->data_object;
int len = 0;
pa_Overflow *overflow;
char *buf = NULL;
if (!doc_data || doc_data->overflow_depth)
return;
/*
* Make sure the doc_data doesn't get removed out from under us.
*/
PA_HoldDocData(doc_data);
/*
** We need to coalesce the overflow buffers into a single one, since
** we don't want to be calling PA_ParseBlock multiple times -- as this could
** possibly get us back into the overflow code.
*/
for (overflow = doc_data->overflow_stack;
overflow != NULL;
overflow = overflow->next)
len += overflow->len;
if (len > 0)
{
buf = (char *)XP_ALLOC_BLOCK((len + 1) * sizeof(char));
buf[0] = 0;
}
while ((overflow = PA_PopOverflow(doc_data)))
{
if (buf != NULL)
XP_STRNCAT(buf, (char*)overflow->buf, overflow->len);
PA_FreeOverflow(overflow);
}
doc_data->overflow_stack = NULL;
if (buf != NULL)
{
PA_ParseBlock(stream, (const char *) buf, len);
XP_FREE(buf);
}
PA_DropDocData(stream);
}
/*
*
*/
static void
pa_unload_complete(NET_StreamClass *stream)
{
pa_DocData *doc_data = (pa_DocData *) stream->data_object;
/* The overflow value was set to one just before sending the JS
onUnload event and the only thing that can change it is
either a document.write or encountering a <SCRIPT> tag, neither
of which should have happened. */
XP_ASSERT(doc_data->overflow_depth == 1);
/* flush any old data that has built up */
pa_FlushOverflow(stream);
}
/*
* If there is a current document see if its OK to unload it now
*/
static void
pa_check_for_new_doc(MWContext *window_id, pa_DocData *doc_data)
{
/* Don't want to do this for layer src changes */
if (doc_data->is_inline_stream)
return;
/* call to see if top_state has unload or we're resizing */
if (LO_CheckForUnload(window_id) ||
doc_data->url_struct->resize_reload)
{
NET_StreamClass *s=NET_NewStream("Place holder",
NULL,
NULL,
NULL,
NULL,
doc_data,
window_id);
/* don't let netlib pass any data up to us yet */
/* doc_data->overflow = 1;*/
PA_PushOverflow(doc_data);
doc_data->overflow_depth ++;
/* send the event to mocha so we get called back */
ET_SendLoadEvent(window_id, EVENT_UNLOAD, (ETVoidPtrFunc) pa_unload_complete,
s, LO_DOCUMENT_LAYER_ID,
doc_data->url_struct->resize_reload);
}
else
{
/*
* Send the event to mocha but we don't need to be
* called back
*/
ET_SendLoadEvent(window_id, EVENT_UNLOAD, NULL, NULL,
LO_DOCUMENT_LAYER_ID,
doc_data->url_struct->resize_reload);
}
}
/*
* Someone has decided they are no longer interested in sending
* data through us. If they were the last ones we were waiting
* on clean everything up now.
*/
pa_DocData *
PA_DropDocData(NET_StreamClass *stream)
{
pa_DocData *doc_data=(pa_DocData *)stream->data_object;
XP_ASSERT(doc_data &&
(doc_data->stream_count >= 1));
if (!doc_data)
return NULL;
doc_data->stream_count--;
return pa_check_doc_data_count(stream);
}
/*
* netlib will no longer send us data
*/
static void
pa_cleanup_after_netlib(NET_StreamClass *stream)
{
pa_DocData *doc_data=stream->data_object;
XP_ASSERT(doc_data);
if (doc_data->window_id)
LO_NetlibComplete(doc_data->window_id);
PA_DropDocData(stream);
}
static void
pa_netlib_stream_complete(NET_StreamClass *stream)
{
pa_cleanup_after_netlib(stream);
}
static void
pa_netlib_stream_abort(NET_StreamClass *stream, int status)
{
pa_DocData *doc_data = (pa_DocData *)stream->data_object;
doc_data->stream_status = status;
pa_cleanup_after_netlib(stream);
}
pa_DocData *
PA_HoldDocData(pa_DocData * doc_data)
{
XP_ASSERT(doc_data);
XP_ASSERT(doc_data->stream_count >= 0);
doc_data->stream_count++;
return doc_data;
}
/*************************************
* Function: PA_BeginParseMDL
*
* Description: The outside world's main access to the parser.
* call this when you are going to start parsing
* a new document to set up the parsing stream.
* This function cannot be called successfully
* until PA_ParserInit() has been called.
*
* Params: Takes lots of document information that is all
* ignored right now, just used the window_id to create
* a unique document id.
*
* Returns: a pointer to a new NET_StreamClass structure, set up to
* give the caller a parsing stream into the parser.
* Returns NULL on error.
*************************************/
NET_StreamClass *
PA_BeginParseMDL(FO_Present_Types format_out,
void *init_data, URL_Struct *anchor, MWContext *window_id)
{
NET_StreamClass *new_stream = NULL;
NET_StreamClass *netlib_stream = NULL;
PA_InitData *new_data;
pa_DocData *doc_data;
new_data = (PA_InitData *)init_data;
/*
* If there was a Window-Target http header from the server,
* we probably need to switch where this document goes.
*/
if ((anchor->window_target != NULL)&&
(*anchor->window_target != '\0')&&
((format_out == FO_PRESENT)||(format_out == FO_CACHE_AND_PRESENT)))
{
if (NET_IsSafeForNewContext(anchor) != FALSE)
{
MWContext *new_context;
/*
* Find the named window if it already exists.
*/
new_context = XP_FindNamedContextInList(window_id,
anchor->window_target);
/*
* If the named window didn't exist, create it.
*/
if (new_context == NULL)
{
/*
* Don't pass in the URL_Struct if we're going to manually
* get it to load ourselves, as MakeNewWindow will load
* a URL passed in!
*/
new_context = FE_MakeNewWindow(window_id, NULL /* anchor */,
anchor->window_target, anchor->window_chrome);
}
/*
* Else is the named window did exist, and we are it, we
* don't need to do anything.
*/
else if (new_context == window_id)
{
new_context = NULL;
}
/*
* Switch to loading this URL in the named window.
*/
if (new_context != NULL)
{
Net_GetUrlExitFunc *exit_func;
exit_func = NULL;
FE_SetWindowLoading(new_context, anchor, &exit_func);
if (NET_SetNewContext(anchor, new_context, exit_func)
== 0)
{
window_id = new_context;
}
}
}
}
if ((anchor != NULL)&&(anchor->address != NULL))
{
char *ret_str;
ret_str = NULL;
(void)HK_CallHook(HK_DOC_START, NULL, XP_CONTEXTID(window_id),
anchor->address, &ret_str);
if (ret_str != NULL)
{
XP_FREE(ret_str);
}
}
/*
* Allocate the crucial data object that contains all the
* per document parsing state information.
*/
doc_data = pa_new_document(format_out, window_id,
new_data->output_func, anchor);
if (doc_data == NULL)
{
return(NULL);
}
/*
* Create the new stream that will stay around even after
* netlib has finished
*/
new_stream = NET_NewStream("netlib stream",
(MKStreamWriteFunc) PA_ParseBlock,
(MKStreamCompleteFunc) PA_MDLComplete,
(MKStreamAbortFunc) PA_MDLAbort,
(MKStreamWriteReadyFunc) pa_ParseWriteReady,
NULL,
window_id);
if (new_stream == NULL)
{
pa_RemoveDocData(doc_data);
if (doc_data->hold_buf)
XP_FREE_BLOCK(doc_data->hold_buf);
if (doc_data->url)
XP_FREE(doc_data->url);
XP_DELETE(doc_data);
return(NULL);
}
/*
* Don't create the separate netlib stream for a stream request
* that's an inline stream that's piggy-backing on some other
* stream (the case if layout_state is non-NULL).
*/
if (!doc_data->is_inline_stream || !doc_data->layout_state)
{
/*
* Create the new stream that netlib is going to free
* out from under us before we are done with it.
*/
netlib_stream = NET_NewStream("netlib stream",
(MKStreamWriteFunc) PA_ParseBlock,
(MKStreamCompleteFunc) pa_netlib_stream_complete,
(MKStreamAbortFunc) pa_netlib_stream_abort,
(MKStreamWriteReadyFunc) pa_ParseWriteReady,
NULL,
window_id);
if (netlib_stream == NULL)
{
pa_RemoveDocData(doc_data);
if (doc_data->hold_buf)
XP_FREE_BLOCK(doc_data->hold_buf);
if (doc_data->url)
XP_FREE(doc_data->url);
XP_DELETE(doc_data);
XP_DELETE(new_stream);
return(NULL);
}
}
else
{
netlib_stream = new_stream;
}
new_stream->data_object = (void *)PA_HoldDocData(doc_data);
netlib_stream->data_object = (void *)doc_data;
/*
* Check to see if its OK to unload the current document (if there
* is one).
*/
pa_check_for_new_doc(window_id, doc_data);
doc_data->format_out = format_out;
doc_data->parser_stream = new_stream;
doc_data->no_newline_count = 0;
doc_data->newline_count = 0;
return(netlib_stream);
}
/*************************************
* Function: PA_ParserInit
*
* Description: Very main interface to the parser library.
* It must be called before the parser can be used.
* Right now it just sets the one static global we use.
* This function can only be called once, all
* subsequent calls will fail.
*
* Params: Pass in a pointer to a PA_Functions struct, which tells the parser
* what functions to use for certain important functionality.
*
* Returns: A status code. 1 on success, -1 on failure.
*************************************/
intn
PA_ParserInit(PA_Functions *funcs)
{
/*
* If this is not the first call, fail them.
*/
if (PA_ParsedTag != NULL)
{
return(-1);
}
PA_ParsedTag = funcs->PA_ParsedTag;
if (PA_ParsedTag == NULL)
{
return(-1);
}
else
{
return(1);
}
}
/*************************************
* Function: PA_ParseBlock
*
* Description: This is a very important entry point to the parser,
* but it will never be called directly. It will be
* placed into the stream returned by PA_BeginParseMDL()
* and be called from there.
*
* Params: The data_object created and placed in the stream class
* in PA_BeginParseMDL(). A buffer of characters to be
* parsed, and the length of that buffer. The buffer is NOT
* a \0 terminated string.
*
* Returns: a status code. 1 = success, -1 = failure.
*************************************/
intn
PA_ParseBlock(NET_StreamClass *stream, const char *block, int block_len)
{
void *data_object=stream->data_object;
pa_DocData *doc_data=data_object;
PA_Tag *tag;
intn ret;
int32 len;
char *buf;
char *hold_buf;
XP_Block buff;
#ifdef XP_WIN16
int32 extra;
char *extra_ptr;
extra_ptr = NULL;
extra = 0;
#endif /* XP_WIN16 */
buf = (char *)block;
len = (int32)block_len;
/*
* Parse this unique MDL document. Get per-document state info.
*/
/*
* If we are overflowing we shouldn't be getting any data
* just glomb it into the overflow buffer
*/
if (doc_data->overflow_depth && len > 0)
{
pa_Overflow *overflow = doc_data->overflow_stack; /* check for NULL? */
char * overflow_buf;
int32 over_size = overflow->size;
int32 over_len = overflow->len;
int32 new_size = over_len + len;
if (over_size < new_size)
{
buff = XP_REALLOC_BLOCK(overflow->buf,
new_size * sizeof(char));
if (buff == NULL)
{
return(MK_OUT_OF_MEMORY);
}
overflow->buf = buff;
overflow->size = new_size;
}
/* lock the buffer so our pointer wizardry works correctly */
XP_LOCK_BLOCK(overflow_buf, char *, overflow->buf);
XP_MEMMOVE(overflow_buf + over_len, buf, len);
XP_UNLOCK_BLOCK(overflow->buf);
overflow->len = new_size;
return 1;
}
/*
* If we are holding buffered data for this document from a
* previous parse attempt,
* Glomb it onto the beginning in a new buffer.
*
* We hold data if we might have a partial MDL tag element.
* We hold data if we might have a partial ampersand escape.
* We hold data if we might have a two character newline to skip.
*/
if (doc_data->hold)
{
if ((doc_data->hold_len + len) > doc_data->hold_size)
{
/*
* Grow the hold buffer if itis not big enough to hold
* the combined buffers.
*/
#ifdef XP_WIN16
/*
* On the 32K limit, our hold_buf is already max size
*/
#else
buff = XP_REALLOC_BLOCK(doc_data->hold_buf,
((doc_data->hold_size + HOLD_BUF_UNIT + len) * sizeof(char)));
if (buff == NULL)
{
return(MK_OUT_OF_MEMORY);
}
doc_data->hold_buf = buff;
doc_data->hold_size += (HOLD_BUF_UNIT + len);
#endif /* XP_WIN16 */
}
/*
* Lock down the hold buffer so we can do pointer magic
* on it.
*/
XP_LOCK_BLOCK(hold_buf, char *, (doc_data->hold_buf));
/*
* Append the new buffer to the old buffer
* Make it look like the merged chunk is what
* was passed to us originally.
*/
#ifdef XP_WIN16
if ((doc_data->hold_len + len) > doc_data->hold_size)
{
extra = doc_data->hold_len + len - doc_data->hold_size;
extra_ptr = (char *)(buf + (len - extra));
XP_BCOPY(buf, (hold_buf + doc_data->hold_len), (len - extra));
buf = hold_buf;
len = doc_data->hold_len + len - extra;
doc_data->hold_len = len;
}
else
{
XP_BCOPY(buf, (hold_buf + doc_data->hold_len), len);
buf = hold_buf;
len = len + doc_data->hold_len;
doc_data->hold_len = len;
}
#else
XP_BCOPY(buf, (hold_buf + doc_data->hold_len), len);
buf = hold_buf;
len = len + doc_data->hold_len;
doc_data->hold_len = len;
#endif /* XP_WIN16 */
}
else
{
/*
* We always want the hold buffer to be locked as we enter the
* following while loop
*/
doc_data->hold_len = 0;
XP_LOCK_BLOCK(hold_buf, char *, (doc_data->hold_buf));
}
doc_data->hold = 0;
/*
* Loop until we get a partial something to hold,
* or we have drained the buffer.
*/
while ((!doc_data->hold)&&(!doc_data->overflow_depth)&&(len != 0))
{
char *tptr;
char *tptr2;
intn is_comment;
/*
* The P_PLAIN_TEXT tag element is very special, and
* if we just got one, all other text is just dumped
* out of the parser as plain text.
*/
if (doc_data->brute_tag == P_PLAIN_TEXT)
{
tag = pa_CreateTextTag(doc_data, buf, len);
if (tag == NULL)
{
return(MK_OUT_OF_MEMORY);
}
ret = doc_data->output_tag(data_object, tag, PA_PARSED);
if (ret < 0)
{
return(ret);
}
buf = NULL;
len = 0;
break;
}
/*
* In the case where we just removed a comment, or just opened
* a preformatting tag, we want to remove any newline that
* appears right after that tag element because we assume
* it was really "part of the element", I.E. WE are guessing
* that the user WANTS us to remove it, a dangerous guess.
*
* Depending on what platform generated the file, a newline
* could be \n, \r, or \r\n. Since it can span 2 characters, we
* may actually have to hold until the next buffer to
* throw away the newline.
*/
if (doc_data->lose_newline != FALSE)
{
if (*buf == '\n')
{
buf++;
len--;
if (len == 0)
{
buf = NULL;
}
if (doc_data->no_newline_count == 0)
doc_data->newline_count++;
doc_data->comment_bytes++;
}
else if ((*buf == '\r')&&(len == 1))
{
doc_data->hold = 1;
/*
* Grow the hold buffer if it is not big enough to hold
* the rest of this buffer.
*/
#ifdef XP_WIN16
/*
* On the 32K limit, our hold_buf is already max size
*/
#else
if (len > doc_data->hold_size)
{
XP_ASSERT(0); /* I don't think we can ever reach
this code since len is always equal to 1 and
hold_size is never zero. */
XP_UNLOCK_BLOCK((doc_data->hold_buf));
buff = XP_REALLOC_BLOCK(doc_data->hold_buf,
((doc_data->hold_size + HOLD_BUF_UNIT + len) *
sizeof(char)));
if (buff == NULL)
{
return(MK_OUT_OF_MEMORY);
}
doc_data->hold_buf = buff;
XP_LOCK_BLOCK(hold_buf, char *,
(doc_data->hold_buf));
doc_data->hold_size += (HOLD_BUF_UNIT + len);
}
#endif /* XP_WIN16 */
XP_BCOPY(buf, hold_buf, len);
doc_data->hold_len = len;
continue;
}
/* Check for Dos or Mac style EOL (need at least 2 chars in buf)*/
else if (*buf == '\r')
{
if ((*((char *)(buf + 1)) == '\n'))
{
/* DOS style EOL */
buf += 2;
len -= 2;
if (len == 0)
{
buf = NULL;
}
if (doc_data->no_newline_count == 0)
doc_data->newline_count++;
doc_data->comment_bytes += 2;
}
else
{
/* Mac style EOL */
buf++;
len--;
if (len == 0)
{
buf = NULL;
}
if (doc_data->no_newline_count == 0)
doc_data->newline_count++;
doc_data->comment_bytes++;
}
}
doc_data->lose_newline = FALSE;
}
/*
* Find the start of any MDL tags in this buffer.
* Returns NULL if there are none.
*/
tptr = pa_FindMDLTag(doc_data, buf, len, &is_comment);
/*
* Some portion of the start of the buffer is text.
*/
if (tptr != buf)
{
int32 text_len;
int32 new_len;
INTL_CharSetInfo c;
/*
* Find the length of the text.
*/
if (tptr == NULL)
{
text_len = len;
}
else
{
text_len = (int32)(tptr - buf);
}
/*
* Expand any ampersand escapes. We might need
* to hold a partial escape. Ampersand escapes
* are NOT expanded if we are inside one of the
* following elements.
*/
if ((doc_data->brute_tag == P_PLAIN_PIECE)||
(doc_data->brute_tag == P_SERVER)||
(doc_data->brute_tag == P_SCRIPT)||
(doc_data->brute_tag == P_STYLE))
{
tptr2 = NULL;
new_len = text_len;
}
else
{
/*
* If we have an MDL tag right after this, we can't have
* partial escapes because the tag is a guaranteed
* terminator. Thus force expansion is set to true.
*/
c = LO_GetDocumentCharacterSetInfo(doc_data->window_id);
if (tptr != NULL)
{
tptr2 = pa_ExpandEscapes(buf, text_len,
&new_len, TRUE, INTL_GetCSIWinCSID(c));
}
else
{
tptr2 = pa_ExpandEscapes(buf, text_len,
&new_len, FALSE,INTL_GetCSIWinCSID(c));
}
/*
* Lump loss of escaped bytes into comment_bytes so that
* dead reckoning in layscrip.c doesn't come up short
*/
if (text_len > new_len)
doc_data->comment_bytes += text_len - new_len;
}
/*
* Create and parse the text into a layout element.
*/
tag = pa_CreateTextTag(doc_data, buf, new_len);
if (tag == NULL)
{
return(MK_OUT_OF_MEMORY);
}
ret = doc_data->output_tag(data_object, tag, PA_PARSED);
if (ret < 0)
{
return(ret);
}
/*
* Check if we are holding a partial ampersand escape.
*/
if (tptr2 != NULL)
{
text_len = (int32)(tptr2 - buf);
tptr = tptr2;
is_comment = COMMENT_MAYBE;
}
/*
* Move up pointers so beginning of tag now heads the buffer.
*/
buf = tptr;
if (buf == NULL)
{
len = 0;
}
else
{
len = len - text_len;
}
if (doc_data->overflow_depth)
{
continue;
}
}
/*
* If we got a maybe, we need to save this
* remnant for later.
*/
if ((is_comment == COMMENT_MAYBE)&&(buf != NULL))
{
doc_data->hold = 1;
/*
* Grow the hold buffer if it is not big enough to hold
* the rest of this buffer.
*/
#ifdef XP_WIN16
/*
* On the 32K limit, our hold_buf is already max size
*/
#else
if (len > doc_data->hold_size)
{
XP_UNLOCK_BLOCK((doc_data->hold_buf));
buff = XP_REALLOC_BLOCK(doc_data->hold_buf,
((doc_data->hold_size + HOLD_BUF_UNIT + len) *
sizeof(char)));
if (buff == NULL)
{
return(MK_OUT_OF_MEMORY);
}
doc_data->hold_buf = buff;
XP_LOCK_BLOCK(hold_buf, char *,
(doc_data->hold_buf));
doc_data->hold_size += (HOLD_BUF_UNIT + len);
}
#endif /* XP_WIN16 */
XP_BCOPY(buf, hold_buf, len);
doc_data->hold_len = len;
}
/*
* else we either have the start of an MDL tag, or
* buf == NULL. Find the end of the tag if we
* have it in the buffer, otherwise return NULL.
*/
else
{
if ((is_comment == COMMENT_YES) || (is_comment == COMMENT_UNCOMMENT))
{
tptr = pa_FindMDLEndComment(doc_data, buf, len);
}
else if (is_comment == COMMENT_PROCESS)
{
tptr = pa_FindMDLEndProcessInstruction(doc_data, buf, len);
}
else
{
tptr = pa_FindMDLEndTag(doc_data, buf, len);
}
/* Skip over comment opening delimiter and JavaScript entity */
if ((tptr != NULL) && (is_comment == COMMENT_UNCOMMENT))
{
int32 skip_len = 0;
/* pa_isolate_javascript_expression() stores a null
character after the JS expresion. */
while (*buf) {
buf++;
skip_len++;
}
buf += 2; /* Skip over }; JS entity delimiter */
skip_len += 2;
/*
* Lump loss of escaped bytes into comment_bytes so that
* dead reckoning in layscrip.c doesn't come up short
*/
doc_data->comment_bytes += skip_len;
len = len - skip_len;
/* Erase the ending comment delimiter */
XP_MEMMOVE(tptr-2, tptr+1, len - (tptr - buf - 3));
len -= 3;
doc_data->comment_bytes += 3;
/* Now start parsing all over again, this time with the comment
delimiters eliminated. */
continue;
}
/*
* Got the end of the MDL comment,
* discard the comment
*/
if ((tptr != NULL)&&((is_comment == COMMENT_YES)||
(is_comment == COMMENT_PROCESS)))
{
int32 comment_len;
comment_len = (int32)(tptr - buf) + 1;
/*
* If we are inside one of the "special"
* tags that ignore all tags except their
* own endtags (e.g. P_TITLE, P_PLAIN_PIECE)
* we need to output this comment
* as normal text.
*/
if (doc_data->brute_tag != P_UNKNOWN)
{
tag = pa_CreateTextTag(doc_data, buf, comment_len);
if (tag == NULL)
{
return(MK_OUT_OF_MEMORY);
}
ret = doc_data->output_tag(data_object, tag, PA_PARSED);
if (ret < 0)
{
return(ret);
}
}
else if (doc_data->window_id && EDT_IS_EDITOR(doc_data->window_id))
{
/*
* The Editor wants to see comments.
*/
tag = pa_CreateMDLTag(doc_data, buf, comment_len);
if (tag == NULL)
{
return(MK_OUT_OF_MEMORY);
}
ret = doc_data->output_tag(data_object, tag, PA_PARSED);
if (ret < 0)
{
return(ret);
}
}
else
{
/*
* Apply lose_newline heuristic after
* discarding a comment.
*/
doc_data->lose_newline = TRUE;
doc_data->comment_bytes += comment_len;
}
/*
* Move the buffer forward.
*/
len = len - comment_len;
if (len == 0)
{
buf = NULL;
}
else
{
buf = tptr;
buf++;
}
}
/*
* Else got the end of the MDL tag!
*/
else if (tptr != NULL)
{
int32 text_len;
/*
* Create and format the tag(s)
*/
text_len = (int32)(tptr - buf) + 1;
tag = pa_CreateMDLTag(doc_data, buf, text_len);
/*
* If we are inside one of the "special"
* tags that ignore all tags except their
* own endtags (e.g. P_TITLE, P_PLAIN_PIECE)
* we check here to see if this is the
* proper end tag, if not, we turn it
* back into normal text.
*/
if (doc_data->brute_tag != P_UNKNOWN)
{
if ((tag == NULL)||
(tag->is_end == FALSE)||
(doc_data->brute_tag != tag->type))
{
PA_FreeTag(tag);
/*
* Strip only the '<' which made us think this
* was an HTML tag.
*/
text_len = 1;
tptr = buf;
tag = pa_CreateTextTag(doc_data, buf, text_len);
if (tag == NULL)
{
return(MK_OUT_OF_MEMORY);
}
}
else
{
doc_data->brute_tag = P_UNKNOWN;
}
}
/*
* These tags are special in that, after opening one
* of them, all other tags are ignored until the matching
* closing tag.
*/
if ((tag != NULL)&&(tag->is_end == FALSE)&&
((tag->type == P_TITLE)||
(tag->type == P_TEXTAREA)||
(tag->type == P_PLAIN_PIECE)||
(tag->type == P_PLAIN_TEXT)||
(tag->type == P_SERVER)||
(tag->type == P_SCRIPT)||
(tag->type == P_STYLE)))
{
doc_data->brute_tag = tag->type;
}
/*
* These tags are special in that, after opening one
* of them, the lose_newline heuristic is applied.
*/
if ((tag != NULL)&&(tag->is_end == FALSE)&&
((tag->type == P_TITLE)||
(tag->type == P_TEXTAREA)||
(tag->type == P_PLAIN_PIECE)||
(tag->type == P_LISTING_TEXT)||
(tag->type == P_PREFORMAT)))
{
doc_data->lose_newline = TRUE;
}
ret = doc_data->output_tag(data_object, tag, PA_PARSED);
if (ret < 0)
{
return(ret);
}
/*
* Move the buffer forward.
*/
len = len - text_len;
if (len == 0)
{
buf = NULL;
}
else
{
buf = tptr;
buf++;
}
if (doc_data->overflow_depth)
{
continue;
}
}
/*
* We couldn't find the end of the MDL tag.
* Hold the start if we have one.
*/
else if (buf != NULL)
{
doc_data->hold = 1;
/*
* Grow the hold buffer if it is not big enough to hold
* the rest of this buffer.
*/
#ifdef XP_WIN16
/*
* On the 32K limit, our hold_buf is already max size
*/
#else
if (len > doc_data->hold_size)
{
XP_UNLOCK_BLOCK((doc_data->hold_buf));
buff = XP_REALLOC_BLOCK(doc_data->hold_buf,
((doc_data->hold_size + HOLD_BUF_UNIT + len) *
sizeof(char)));
if (buff == NULL)
{
return(MK_OUT_OF_MEMORY);
}
doc_data->hold_buf = buff;
XP_LOCK_BLOCK(hold_buf, char *,
(doc_data->hold_buf));
doc_data->hold_size += (HOLD_BUF_UNIT + len);
}
#endif /* XP_WIN16 */
XP_BCOPY(buf, hold_buf, len);
doc_data->hold_len = len;
}
} /* end of else on COMMENT_MAYBE */
} /* end of while */
/*
* If we got here because we entered overflow mode all of the
* left over data should go into the overflow buffer and the
* hold buffer should be left empty
*/
if (doc_data->overflow_depth && len > 0)
{
pa_Overflow *overflow = doc_data->overflow_stack; /* check for NULL? */
char * overflow_buf;
int32 over_len = overflow->len;
int32 over_size = overflow->size;
int32 new_size = over_len + len;
XP_UNLOCK_BLOCK((doc_data->hold_buf));
if (over_size < new_size)
{
buff = XP_REALLOC_BLOCK(overflow->buf,
new_size * sizeof(char));
if (buff == NULL)
{
return(MK_OUT_OF_MEMORY);
}
overflow->buf = buff;
overflow->size = new_size;
}
/*
* We could be processing data that was passed in from the
* overflow_buf so we need to do an over-lap safe copy here
*/
XP_LOCK_BLOCK(overflow_buf, char *, overflow->buf);
XP_MEMMOVE(overflow_buf + over_len, buf, len);
XP_UNLOCK_BLOCK(overflow->buf);
overflow->len = new_size;
return 1;
}
/*
* Unlock the hold buffer, and clear it if we
* aren't holding anything this time around.
*/
if ((!doc_data->hold)&&(doc_data->hold_buf != NULL))
{
hold_buf = NULL;
doc_data->hold_len = 0;
XP_UNLOCK_BLOCK((doc_data->hold_buf));
}
else if (doc_data->hold_buf != NULL)
{
XP_UNLOCK_BLOCK((doc_data->hold_buf));
}
#ifdef XP_WIN16
if (extra_ptr != NULL)
{
if ((doc_data->hold_len + extra) > doc_data->hold_size)
{
char minibuf[1];
intn ret;
XP_LOCK_BLOCK(hold_buf, char *, (doc_data->hold_buf));
minibuf[0] = *hold_buf;
XP_BCOPY((char *)(hold_buf + 1), hold_buf, (doc_data->hold_len - 1));
doc_data->hold_len--;
XP_UNLOCK_BLOCK((doc_data->hold_buf));
tag = pa_CreateTextTag(doc_data, minibuf, 1);
if (tag == NULL)
{
return(MK_OUT_OF_MEMORY);
}
ret = doc_data->output_tag(data_object, tag, PA_PARSED);
if (ret < 0)
{
return(ret);
}
ret = PA_ParseBlock(stream, extra_ptr, extra);
return(ret);
}
else
{
XP_LOCK_BLOCK(hold_buf, char *, (doc_data->hold_buf));
XP_BCOPY(extra_ptr, (char *)(hold_buf + doc_data->hold_len), extra);
doc_data->hold_len += extra;
XP_UNLOCK_BLOCK((doc_data->hold_buf));
}
}
#endif /* XP_WIN16 */
return(1);
}
/*************************************
* Function: PA_MDLComplete
*
* Description: This is a very important entry point to the parser,
* but it will never be called directly. It will be
* placed into the stream returned by PA_BeginParseMDL()
* and be called from there. It tells the parser that the
* passed document is done parsing, there is no new data.
*
* Params: The data_object created and placed in the stream class
* in PA_BeginParseMDL(). This contains document specific
* parse state information.
*
* Returns: nothing.
*************************************/
void
PA_MDLComplete(NET_StreamClass *stream)
{
void *data_object=stream->data_object;
pa_DocData *doc_data;
PA_Tag *tag;
Bool is_main_context_stream;
/*
* This MDL document is complete
*/
doc_data = (pa_DocData *)data_object;
XP_ASSERT(doc_data->overflow_depth == 0);
/*
* If we were holding some data we hadn't parsed yet, we need
* to flush it through now. Since we couldn't resolve whatever
* we were waiting for, just push it through as plain text.
*/
if (doc_data->hold)
{
char *tptr;
/*
* Lock down the hold buffer so we can do pointer magic
* on it.
*/
XP_LOCK_BLOCK(tptr, char *, (doc_data->hold_buf));
/*
* Due to NCSA Mosaic, there are many incorrectly
* commented documents out there, that have the <!-- comment
* start, but expect a different end such as --!> or just >
* If we have finished this document, and the hold buffer
* has a starting comment, this is probably what happened,
* So terminate the starting comment with the next > and
* parse on.
* Of necessity this may involve recursion for multiple comments
*/
if ((doc_data->hold_len > 4)&&(XP_STRNCMP(tptr, "<!--", 4) == 0))
{
char *nothing;
NET_StreamClass *newstream;
/*
* Break the comment
*/
tptr[2] = 'C';
XP_UNLOCK_BLOCK((doc_data->hold_buf));
/*
* Reparse this, then call yourself.
*/
nothing = (char*) XP_ALLOC(1);
if (nothing == NULL)
{
XP_FREE_BLOCK((doc_data->hold_buf));
return;
}
nothing[0] = '\0';
newstream = doc_data->parser_stream;
PA_HoldDocData(doc_data);
(void)PA_ParseBlock(newstream, nothing, 0);
XP_FREE(nothing);
PA_DropDocData(newstream);
return;
}
#ifndef LENIENT_END_TAG
/*
* Also due to allowing '>' in quoted attributes, if they forget
* to close a quote, and there is no other in the entire document,
* we could have held the whole thing looking for the close quote,
* and never found the tag. Skip this malformed tag by dropping
* its starting '<' and parsing on.
* Of necessity this may involve recursion for multiple errors.
*/
else if ((doc_data->hold_len > 3)&&(*tptr == '<'))
{
char *nothing;
PA_Tag *tmp_tag;
NET_StreamClass *newstream;
/*
* Push out the '<'
*/
nothing = (char*) XP_ALLOC(1);
if (nothing == NULL)
{
XP_FREE_BLOCK((doc_data->hold_buf));
return;
}
nothing[0] = '<';
tmp_tag = pa_CreateTextTag(doc_data, nothing, 1);
doc_data->output_tag(data_object, tmp_tag, PA_PARSED);
/*
* Remove the '<' an move up the hold buffer.
*/
XP_BCOPY((char *)(tptr + 1), tptr, (doc_data->hold_len - 1));
doc_data->hold_len--;
XP_UNLOCK_BLOCK((doc_data->hold_buf));
/*
* reparse the rest.
*/
nothing[0] = '\0';
newstream = doc_data->parser_stream;
PA_HoldDocData(doc_data);
(void)PA_ParseBlock(newstream, nothing, 0);
XP_FREE(nothing);
PA_DropDocData(newstream);
return;
}
#endif /* LENIENT_END_TAG */
tag = pa_CreateTextTag(doc_data, tptr, doc_data->hold_len);
doc_data->output_tag(data_object, tag, PA_PARSED);
XP_UNLOCK_BLOCK((doc_data->hold_buf));
XP_FREE_BLOCK((doc_data->hold_buf));
doc_data->hold_len = 0;
doc_data->hold_buf = NULL;
doc_data->hold = 0;
}
else if (doc_data->hold_buf != NULL)
{
XP_FREE_BLOCK((doc_data->hold_buf));
doc_data->hold_buf = NULL;
doc_data->hold_size = 0;
}
doc_data->output_tag(data_object, NULL, PA_COMPLETE);
is_main_context_stream = pa_RemoveDocData(doc_data);
/*
* free up all the data allocated when this stream was initiated.
*/
if (doc_data->url_struct != NULL)
{
NET_FreeURLStruct(doc_data->url_struct);
}
if (doc_data->overflow_stack)
{
pa_Overflow *overflow;
while ((overflow = PA_PopOverflow(doc_data)))
PA_FreeOverflow(overflow);
doc_data->overflow_stack = NULL;
}
if (doc_data->url != NULL)
{
XP_FREE(doc_data->url);
}
if (is_main_context_stream)
{
XP_DELETE(doc_data->parser_stream);
}
XP_DELETE(doc_data);
}
/*************************************
* Function: PA_MDLAbort
*
* Description: This is a very important entry point to the parser,
* but it will never be called directly. It will be
* placed into the stream returned by PA_BeginParseMDL()
* and be called from there. It tells the parser that the
* passed document is aborted, there is no new data,
* throw out everything you have and stop parsing.
*
* Params: The data_object created and placed in the stream class
* in PA_BeginParseMDL(). This contains document specific
* parse state information. Also passed a character message
* which is the reason for the abort.
*
* Returns: nothing.
*************************************/
void
PA_MDLAbort(NET_StreamClass *stream, int status)
{
void *data_object=stream->data_object;
pa_DocData *doc_data;
Bool is_main_context_stream;
/*
* This MDL document is complete
*/
doc_data = (pa_DocData *)data_object;
/*
* If we were holding some data we hadn't parsed yet, we need
* to throw it out now.
*/
if (doc_data->hold)
{
XP_UNLOCK_BLOCK((doc_data->hold_buf));
XP_FREE_BLOCK((doc_data->hold_buf));
doc_data->hold_len = 0;
doc_data->hold_buf = NULL;
doc_data->hold = 0;
}
else if (doc_data->hold_buf != NULL)
{
XP_FREE_BLOCK((doc_data->hold_buf));
doc_data->hold_buf = NULL;
doc_data->hold_size = 0;
}
doc_data->output_tag(data_object, NULL, PA_ABORT);
is_main_context_stream = pa_RemoveDocData(doc_data);
/*
* free up all the data allocated when this stream was initiated.
*/
if (doc_data->url_struct != NULL)
{
NET_FreeURLStruct(doc_data->url_struct);
doc_data->url_struct = NULL;
}
if (doc_data->url != NULL)
{
XP_FREE(doc_data->url);
}
if (doc_data->overflow_stack)
{
pa_Overflow *overflow;
while ((overflow = PA_PopOverflow(doc_data)))
PA_FreeOverflow(overflow);
}
doc_data->overflow_depth = 0;
if (is_main_context_stream)
{
XP_DELETE(doc_data->parser_stream);
}
XP_DELETE(doc_data);
}
static pa_DocData *
pa_FetchDocData(MWContext *window_id)
{
pa_DocDataList *dptr;
pa_DocData *doc_data;
dptr = DocDataList;
while (dptr != NULL)
{
if (dptr->window_id == window_id)
{
break;
}
dptr = dptr->next;
}
if (dptr == NULL)
{
doc_data = NULL;
}
else
{
doc_data = dptr->doc_data;
}
return(doc_data);
}
XP_Bool ValidateDocData(MWContext *window_id)
{
if (pa_FetchDocData(window_id))
return TRUE;
return FALSE;
}
static Bool
pa_RemoveDocData(pa_DocData *target_doc_data)
{
pa_DocDataList *dptr;
pa_DocDataList *doc_data;
doc_data = DocDataList;
dptr = DocDataList;
while (dptr != NULL)
{
if (dptr->doc_data == target_doc_data)
{
break;
}
doc_data = dptr;
dptr = dptr->next;
}
if (dptr != NULL)
{
if (dptr == DocDataList)
{
DocDataList = DocDataList->next;
}
else
{
doc_data->next = dptr->next;
}
XP_DELETE(dptr);
return TRUE;
}
return FALSE;
}
static Bool
pa_StoreDocData(MWContext *window_id, pa_DocData *new_doc_data)
{
pa_DocDataList *dptr;
dptr = DocDataList;
while (dptr != NULL)
{
if (dptr->window_id == window_id)
{
break;
}
dptr = dptr->next;
}
if (dptr == NULL)
{
dptr = XP_NEW(pa_DocDataList);
if (dptr == NULL)
{
return FALSE;
}
dptr->window_id = window_id;
dptr->next = DocDataList;
DocDataList = dptr;
}
dptr->doc_data = new_doc_data;
return TRUE;
}
void
PA_PushOverflow(pa_DocData* doc_data)
{
pa_Overflow *overflow = XP_NEW(pa_Overflow);
if (overflow == NULL) return;
overflow->buf = XP_ALLOC(HOLD_BUF_UNIT * sizeof(char));
if (overflow->buf == NULL)
{
XP_FREE(overflow);
return;
}
overflow->buf[0] = 0;
overflow->size = HOLD_BUF_UNIT;
overflow->len = 0;
overflow->next = doc_data->overflow_stack;
doc_data->overflow_stack = overflow;
}
pa_Overflow *
PA_PopOverflow(pa_DocData* doc_data)
{
pa_Overflow *overflow = doc_data->overflow_stack;
if (!overflow) return NULL;
doc_data->overflow_stack = overflow->next;
overflow->next = NULL;
return overflow;
}
void
PA_FreeOverflow(pa_Overflow *overflow)
{
XP_FREE(overflow->buf);
XP_FREE(overflow);
}
int
PA_GetOverflowDepth(pa_DocData* doc_data)
{
return doc_data->overflow_depth;
}
XP_Block
PA_GetOverflowBuf(pa_DocData* doc_data)
{
XP_ASSERT(doc_data->overflow_stack);
if (!doc_data->overflow_stack) return NULL;
return doc_data->overflow_stack->buf;
}
#ifdef PROFILE
#pragma profile off
#endif