From mboxrd@z Thu Jan 1 00:00:00 1970 Path: news.gmane.org!not-for-mail From: Michael Welsh Duggan Newsgroups: gmane.emacs.bugs Subject: bug#11749: Acknowledgement (24.1; C-mode indentation gives wrong-type-argument error.) Date: Mon, 10 Sep 2012 09:10:27 -0400 Message-ID: References: <87ehpbni8j.fsf@kfs-lx2.home.kfs> <503D4B11.6000501@cua.dk> <20120902211620.GA3824@acm.acm> <50447C94.2040402@cua.dk> <20120905204821.GA3620@acm.acm> <87ipbqpkb7.fsf@maru.md5i.com> <20120908211451.GA22477@acm.acm> NNTP-Posting-Host: plane.gmane.org Mime-Version: 1.0 Content-Type: multipart/mixed; boundary="=-=-=" X-Trace: ger.gmane.org 1347282723 4229 80.91.229.3 (10 Sep 2012 13:12:03 GMT) X-Complaints-To: usenet@ger.gmane.org NNTP-Posting-Date: Mon, 10 Sep 2012 13:12:03 +0000 (UTC) Cc: 11749@debbugs.gnu.org, Kim Storm To: Alan Mackenzie Original-X-From: bug-gnu-emacs-bounces+geb-bug-gnu-emacs=m.gmane.org@gnu.org Mon Sep 10 15:12:05 2012 Return-path: Envelope-to: geb-bug-gnu-emacs@m.gmane.org Original-Received: from lists.gnu.org ([208.118.235.17]) by plane.gmane.org with esmtp (Exim 4.69) (envelope-from ) id 1TB3mW-00077L-J1 for geb-bug-gnu-emacs@m.gmane.org; Mon, 10 Sep 2012 15:12:04 +0200 Original-Received: from localhost ([::1]:57436 helo=lists.gnu.org) by lists.gnu.org with esmtp (Exim 4.71) (envelope-from ) id 1TB3mT-0000S6-0a for geb-bug-gnu-emacs@m.gmane.org; Mon, 10 Sep 2012 09:12:01 -0400 Original-Received: from eggs.gnu.org ([208.118.235.92]:60363) by lists.gnu.org with esmtp (Exim 4.71) (envelope-from ) id 1TB3mH-0000QO-D7 for bug-gnu-emacs@gnu.org; Mon, 10 Sep 2012 09:11:58 -0400 Original-Received: from Debian-exim by eggs.gnu.org with spam-scanned (Exim 4.71) (envelope-from ) id 1TB3m5-0006qN-Iq for bug-gnu-emacs@gnu.org; Mon, 10 Sep 2012 09:11:49 -0400 Original-Received: from debbugs.gnu.org ([140.186.70.43]:42059) by eggs.gnu.org with esmtp (Exim 4.71) (envelope-from ) id 1TB3lw-0006ou-G7; Mon, 10 Sep 2012 09:11:28 -0400 Original-Received: from Debian-debbugs by debbugs.gnu.org with local (Exim 4.72) (envelope-from ) id 1TB3mU-0006oj-Cx; Mon, 10 Sep 2012 09:12:02 -0400 X-Loop: help-debbugs@gnu.org Resent-From: Michael Welsh Duggan Original-Sender: debbugs-submit-bounces@debbugs.gnu.org Resent-CC: bug-gnu-emacs@gnu.org, bug-cc-mode@gnu.org Resent-Date: Mon, 10 Sep 2012 13:12:02 +0000 Resent-Message-ID: Resent-Sender: help-debbugs@gnu.org X-GNU-PR-Message: followup 11749 X-GNU-PR-Package: emacs,cc-mode X-GNU-PR-Keywords: Original-Received: via spool by 11749-submit@debbugs.gnu.org id=B11749.134728267126021 (code B ref 11749); Mon, 10 Sep 2012 13:12:02 +0000 Original-Received: (at 11749) by debbugs.gnu.org; 10 Sep 2012 13:11:11 +0000 Original-Received: from localhost ([127.0.0.1]:51604 helo=debbugs.gnu.org) by debbugs.gnu.org with esmtp (Exim 4.72) (envelope-from ) id 1TB3ld-0006lX-VF for submit@debbugs.gnu.org; Mon, 10 Sep 2012 09:11:11 -0400 Original-Received: from euclid.red.cert.org ([192.88.209.48]:57899) by debbugs.gnu.org with esmtp (Exim 4.72) (envelope-from ) id 1TB3lZ-0006lG-G4 for 11749@debbugs.gnu.org; Mon, 10 Sep 2012 09:11:07 -0400 Original-Received: from bucknell.indigo.cert.org (bucknell.indigo.cert.org [10.60.10.121]) by euclid.red.cert.org (8.14.4/8.14.4) with ESMTP id q8AD4WqC004342; Mon, 10 Sep 2012 09:04:32 -0400 DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=cert.org; s=jthatj15xw2j; t=1347282272; bh=xrUJFSD/uFyRbCMZGYSwTNVXSE+NxeVifyPV0tb5AcQ=; h=From:To:Cc:Subject:References:Date:In-Reply-To:Message-ID: MIME-Version:Content-Type:Sender:Reply-To; b=b2+W/4URagNmKdl5Zg4GIu0FrIfS/LrkiXHZoqtO5LFpO5m2Fu90CeWMN4T/C9r/I P6V168apzzAFCSqCxFT/xl0YNwkgCxvrPFVgx+uJy3OyF1a9cLCkIO/wzDbPSEyvfS UBhuCsV1svUBFjlxJj3xxwc1Bif5vomR0/nPwEQI= Original-Received: from waterbuck.yellow.cert.org (waterbuck.yellow.cert.org [10.20.128.84]) by bucknell.indigo.cert.org (8.14.4/8.14.4/2.81) with ESMTP id q8ADAS50013865; Mon, 10 Sep 2012 09:10:28 -0400 Original-Received: (from mwd@localhost) by waterbuck.yellow.cert.org (8.14.4/8.14.4/Submit/1.6) id q8ADARK9025459; Mon, 10 Sep 2012 09:10:27 -0400 X-Authentication-Warning: waterbuck.yellow.cert.org: mwd set sender to mwd@cert.org using -f In-Reply-To: (Michael Welsh Duggan's message of "Mon, 10 Sep 2012 08:18:51 -0400") User-Agent: Gnus/5.130006 (Ma Gnus v0.6) Emacs/24.1.50 (gnu/linux) X-BeenThere: debbugs-submit@debbugs.gnu.org X-Mailman-Version: 2.1.13 Precedence: list X-detected-operating-system: by eggs.gnu.org: GNU/Linux 2.6 (newer, 2) X-Received-From: 140.186.70.43 X-BeenThere: bug-gnu-emacs@gnu.org List-Id: "Bug reports for GNU Emacs, the Swiss army knife of text editors" List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , Errors-To: bug-gnu-emacs-bounces+geb-bug-gnu-emacs=m.gmane.org@gnu.org Original-Sender: bug-gnu-emacs-bounces+geb-bug-gnu-emacs=m.gmane.org@gnu.org Xref: news.gmane.org gmane.emacs.bugs:64037 Archived-At: --=-=-= Content-Type: text/plain Michael Welsh Duggan writes: >>> There is generally no way to re-create it, so I've stopped reporting >>> these. I wish there were some way to record all actions in c-mode >>> buffers such that they could be saved and re-played when this type of >>> problem happens. If there were some sort of debug flag I could turn >>> on, I would turn it on by default and hopefully be able to catch some >>> useful information. >> >> M-x c-toggle-parse-state-debug >> >> (or (c-toggle-parse-state-debug 1) in your .emacs). Warning: variable >> `c-debug-parse-state' is not buffer local. I wrote this on 19th October >> last year to help sort out the bug you reported a little earlier. :-) > > I will turn this on. Okay, I have something repeatable. This is with an older checkout ("109494 dmantipov@yandex.ru-20120807112841-k0pyiinoxi2llcmu"), so I'm updating my sources to see if this holds in the current bzr. In the meantime, here's a recipe that works in the above version. 1) emacs -Q 2) C-x C-f 3) M-x c-toggle-parse-state-debug RET 4) C-v until you get a state inconsistency message (10 times for me) c-parse-state inconsistency at 9942: using cache: (9866 9516 5134 5090), from scratch: (9866 9516 (7654 . 9439) 5134 5090) Old state: (setq c-state-cache '(9516 5134 5090) c-state-cache-good-pos 10231 c-state-nonlit-pos-cache '(15063 12049 9049 6049 3049) c-state-nonlit-pos-cache-limit 15063 c-state-brace-pair-desert '(9516 . 9818) c-state-point-min 1 c-state-point-min-lit-type nil c-state-point-min-lit-start nil c-state-min-scan-pos 1 c-state-old-cpp-beg nil c-state-old-cpp-end nil) c-parse-state inconsistency at 9942: using cache: (9866 9516 5134 5090), from scratch: (9866 9516 (7654 . 9439) 5134 5090) Old state: (setq c-state-cache '(9866 9516 5134 5090) c-state-cache-good-pos 9867 c-state-nonlit-pos-cache '(15063 12049 9049 6049 3049) c-state-nonlit-pos-cache-limit 15063 c-state-brace-pair-desert '(9516 . 9818) c-state-point-min 1 c-state-point-min-lit-type nil c-state-point-min-lit-start nil c-state-min-scan-pos 1 c-state-old-cpp-beg nil c-state-old-cpp-end nil) --=-=-= Content-Type: text/plain Content-Disposition: attachment; filename=conn_memory.lzz /* ** Copyright (C) 2012 by Carnegie Mellon University. ** ** @OPENSOURCE_HEADER_START@ ** ** Use of the SILK system and related source code is subject to the terms ** of the following licenses: ** ** GNU Public License (GPL) Rights pursuant to Version 2, June 1991 ** Government Purpose License Rights (GPLR) pursuant to DFARS 252.227.7013 ** ** NO WARRANTY ** ** ANY INFORMATION, MATERIALS, SERVICES, INTELLECTUAL PROPERTY OR OTHER ** PROPERTY OR RIGHTS GRANTED OR PROVIDED BY CARNEGIE MELLON UNIVERSITY ** PURSUANT TO THIS LICENSE (HEREINAFTER THE "DELIVERABLES") ARE ON AN ** "AS-IS" BASIS. CARNEGIE MELLON UNIVERSITY MAKES NO WARRANTIES OF ANY ** KIND, EITHER EXPRESS OR IMPLIED AS TO ANY MATTER INCLUDING, BUT NOT ** LIMITED TO, WARRANTY OF FITNESS FOR A PARTICULAR PURPOSE, ** MERCHANTABILITY, INFORMATIONAL CONTENT, NONINFRINGEMENT, OR ERROR-FREE ** OPERATION. CARNEGIE MELLON UNIVERSITY SHALL NOT BE LIABLE FOR INDIRECT, ** SPECIAL OR CONSEQUENTIAL DAMAGES, SUCH AS LOSS OF PROFITS OR INABILITY ** TO USE SAID INTELLECTUAL PROPERTY, UNDER THIS LICENSE, REGARDLESS OF ** WHETHER SUCH PARTY WAS AWARE OF THE POSSIBILITY OF SUCH DAMAGES. ** LICENSEE AGREES THAT IT WILL NOT MAKE ANY WARRANTY ON BEHALF OF ** CARNEGIE MELLON UNIVERSITY, EXPRESS OR IMPLIED, TO ANY PERSON ** CONCERNING THE APPLICATION OF OR THE RESULTS TO BE OBTAINED WITH THE ** DELIVERABLES UNDER THIS LICENSE. ** ** Licensee hereby agrees to defend, indemnify, and hold harmless Carnegie ** Mellon University, its trustees, officers, employees, and agents from ** all claims or demands made against them (and any related losses, ** expenses, or attorney's fees) arising out of, or relating to Licensee's ** and/or its sub licensees' negligent use or willful misuse of or ** negligent conduct or willful misconduct regarding the Software, ** facilities, or other rights or assistance granted by Carnegie Mellon ** University under this License, including, but not limited to, any ** claims of product liability, personal injury, death, damage to ** property, or violation of any laws or regulations. ** ** Carnegie Mellon University Software Engineering Institute authored ** documents are sponsored by the U.S. Department of Defense under ** Contract FA8721-05-C-0003. Carnegie Mellon University retains ** copyrights in all material produced under this contract. The U.S. ** Government retains a non-exclusive, royalty-free license to publish or ** reproduce these documents, or allow others to do so, for U.S. ** Government purposes only pursuant to the copyright license under the ** contract clause at 252.227.7013. ** ** @OPENSOURCE_HEADER_END@ */ #hdr #include RCSIDENTVAR(rcsID_CONN_MEMORY_H, "$Id$"); #include "connector.hh" #end #src RCSIDENT("$Id$"); #include #ifdef SKCONNECTOR_TRACE_LEVEL #define TRACEMSG_LEVEL SKCONNECTOR_TRACE_LEVEL #endif #define TRACEMSG(lvl, msg) TRACEMSG_TO_TRACEMSGLVL(lvl, msg) #include /* * * Tracing macros * */ /* Levels at which certain types of tracing are emitted */ #define FUNCTION_LEVEL 3 /* Main function tracing */ #define CHUNK_LEVEL 4 /* Chunk tracing */ /* * TRACE_FUNC; * * Emits a trace message for the containing function. The pointer * emitted is the connection object. */ #define TRACE_FUNC \ TRACEMSG(FUNCTION_LEVEL, ("Function: (%p) %s", conn, __func__)) /* * TRACE_MEM_CHUNK(chunk); * * Emits trace info for a conn_mem_chunk_t data structure. */ #define TRACE_MEM_CHUNK(chunk) \ TRACEMSG(CHUNK_LEVEL, \ (("%s:%d Chunk: %p, writer_pos: %zd" \ " reader_pos: %zd max_reader_pos: %zd"), \ __FILE__, __LINE__, \ chunk, chunk->writer_pos, chunk->reader_pos, \ chunk->max_reader_pos)); /* TYPEDEFS AND DEFINES */ /* minimum size we allow for a chunk is 4k */ #define CONN_MEM_MIN_CHUNK_SIZE 0x1000 /* standard chunk size is half a meg */ #define CONN_MEM_STD_CHUNK_SIZE 0x80000 /* standard number of chunks the connector norally has---used to * compute the maximum allocation */ #define CONN_MEM_STD_NUMBER_CHUNKS 3 /* * * Macros common to all connectors * */ /* when the writer wraps, ensure this much space between the end of * the writer's block and the start of the reader's block */ #define WRAP_GAP sizeof(uint64_t) /* total space required for a block that holds 'bts_size' bytes of * memory in its 'data' member; includes the 'block_size' member and * ensures 'data' is 64-bit aligned */ #define BLOCK_TOTAL_SIZE(bts_size) \ (((bts_size) + 2 * sizeof(uint64_t) - 1) & ~(UINT64_C(7))) /* max size a caller may request; we require at least 3 blocks per * chunk; allow for overhead within the chunk */ #define BLOCK_MAX_SIZE_FROM_CHUNK_SIZE(bms_chunk_size) \ (((bms_chunk_size) - 4 * WRAP_GAP) / 3) #end namespace silk { class MemoryConnector : public Connector { private: struct conn_mem_chunk_t { /* pointer to the next chunk */ conn_mem_chunk_t *next; /* the data used for conn_block_t, which 'writer_pos' and * 'reader_pos' are offsets into */ uint8_t *blocks; /* offset to the location for the upstream writer */ size_t writer_pos; /* offset to the location for the downstream reader */ size_t reader_pos; /* total number of bytes in 'blocks'; this space includes blocks * the callers access and any overhead */ size_t capacity; /* last valid byte that can be read in 'blocks'; used to determine * when the reader_pos needs to be reset to 0 */ size_t max_reader_pos; }; public: const char *get_type() const { return CONN_MEM_NAME; } private: static void setup_class(); static void teardown_class( void *dummy); MemoryConnector(); virtual ~MemoryConnector(); conn_mem_chunk_t *chunk_alloc() { conn_mem_chunk_t *chunk; chunk = new conn_mem_chunk_t; chunk->blocks = new uint8_t[chunk_size]; chunk->capacity = chunk_size; return chunk; } void chunk_free( conn_mem_chunk_t *chunk) { if (chunk) { if (chunk->blocks) { delete[] chunk->blocks; } delete chunk; } } void chunk_pop() { conn_mem_chunk_t *chunk; assert(reader_chunk); assert(reader_chunk->next); if (!spare_chunk) { spare_chunk = reader_chunk; reader_chunk = reader_chunk->next; } else { chunk = reader_chunk; reader_chunk = reader_chunk->next; chunk_free(chunk); } } void chunk_push() { if (spare_chunk) { spare_chunk->next = NULL; spare_chunk->writer_pos = 0; spare_chunk->reader_pos = 0; spare_chunk->max_reader_pos = 0; writer_chunk->next = spare_chunk; writer_chunk = writer_chunk->next; spare_chunk = NULL; } else { writer_chunk->next = chunk_alloc(); writer_chunk = writer_chunk->next; } } int _get_read_block( const conn_block_t **reader_block, size_t size, bool no_wait) { conn_mem_chunk_t *chunk; assert(reader_block); assert(size); for (;;) { chunk = reader_chunk; if (chunk) { if (stopped) { return -1; } if (chunk->reader_pos != chunk->writer_pos) { if (chunk->max_reader_pos && (chunk->reader_pos == chunk->max_reader_pos)) { /* wrap the reader if the writer has wrapped */ chunk->reader_pos = 0; chunk->max_reader_pos = 0; continue; } /* there is data to return */ break; } /* else, there is no data in this block */ if (chunk->next) { /* I don't think this ever gets called, since we * should have moved to the next block when the reader * returned its previous block */ /* free this block and move to the next */ chunk_pop(); continue; } } /* else no data is available */ /* check the stopping condition */ if (stopped || no_more_data) { return -1; } /* wait for data unless no_wait was specified */ empty = true; if (no_wait) { return 1; } pthread_cond_wait(&cond, &mutex); } TRACE_FUNC; /* we get here when there is data to return */ *reader_block = (conn_block_t*)&chunk->blocks[chunk->reader_pos]; return 0; } void _return_read_block( const conn_block_t *reader_block) { conn_mem_chunk_t *chunk; assert(reader_block); chunk = reader_chunk; assert(reader_block==(conn_block_t*)&chunk->blocks[chunk->reader_pos]); /* move to next reader position */ chunk->reader_pos += BLOCK_TOTAL_SIZE(reader_block->block_size); if (chunk->reader_pos == chunk->max_reader_pos) { chunk->reader_pos = 0; chunk->max_reader_pos = 0; } TRACE_MEM_CHUNK(chunk); /* check whether we need to move to the next chunk */ if (chunk->reader_pos == chunk->writer_pos && chunk->next) { /* free this block and move to the next */ chunk_pop(); } } int _get_write_block( const conn_block_t **writer_block, size_t size, bool no_wait) { conn_mem_chunk_t *chunk; size_t required_size; assert(writer_block); assert(size); *writer_block = NULL; if (block_max_size < size) { throw sk_error_conn_block_size_too_large_t(); } /* account for overhead */ required_size = BLOCK_TOTAL_SIZE(size); /* loop until there is space available */ for (;;) { /* handle stopped condition */ if (stopped) { return -1; } if (total_used + size > max_allocation) { /* we are full; wait for space unless no_wait was * specified */ full = true; if (no_wait) { return 1; } pthread_cond_wait(&cond, &mutex); continue; } chunk = writer_chunk; if (NULL == chunk) { /* create the initial chunk */ chunk = writer_chunk = reader_chunk = chunk_alloc(); } if (chunk->writer_pos < chunk->reader_pos) { /* the writer has wrapped; look for space between the * writer_pos and the reader_pos */ if ((chunk->reader_pos - chunk->writer_pos) >= (required_size + WRAP_GAP)) { /* space is available */ break; } } else { if ((chunk->capacity - chunk->writer_pos) >= required_size) { /* space is available at end of the buffer */ break; } /* else, see if space is available at the front */ if (chunk->reader_pos >= required_size + WRAP_GAP) { /* space is available; wrap around */ chunk->max_reader_pos = chunk->writer_pos; chunk->writer_pos = 0; TRACE_MEM_CHUNK(chunk); break; } } /* space is not available; allocate new block */ chunk_push(); } TRACE_FUNC; /* when we get here, space is available */ *writer_block = (conn_block_t*)&chunk->blocks[chunk->writer_pos]; ((conn_block_t*)&chunk->blocks[chunk->writer_pos])->block_size = required_size - sizeof(uint64_t); return 0; } void _return_write_block( const conn_block_t *writer_block, size_t size) { conn_mem_chunk_t *chunk; assert(writer_block); assert(size > 0); chunk = writer_chunk; assert(writer_block==(conn_block_t*)&chunk->blocks[chunk->writer_pos]); ((conn_block_t*)&chunk->blocks[chunk->writer_pos])->block_size = size; chunk->writer_pos += BLOCK_TOTAL_SIZE(size); TRACE_MEM_CHUNK(chunk); } void _set_param( const Definition& defn, const std::vector& values) throw (Error) { uint64_t u64; int rv; switch (*(param_id_t *)(defn.get_context())) { case CONN_MEM_CHUNK_SIZE: rv = skStringParseUint64(&u64, values[0].c_str(), 0, 0); if (rv) { goto PARSE_ERROR; } chunk_size = u64; break; case CONN_MEM_MAX_ALLOCATION: rv = skStringParseUint64(&u64, values[0].c_str(), 0, 0); if (rv) { goto PARSE_ERROR; } max_allocation = u64; break; } return; PARSE_ERROR: std::ostringstream os; os << "Invalid " << defn.get_name() << " '" << values[0] << "': " << skStringParseStrerror(rv); throw sk_error_conn_param_value_t(os.str()); } void _init() { if (chunk_size) { if (chunk_size < CONN_MEM_MIN_CHUNK_SIZE) { std::ostringstream os; os << "Specified chunk size is below minimum of " << CONN_MEM_MIN_CHUNK_SIZE; throw sk_error_conn_param_value_t(os.str()); } if (0 == max_allocation) { /* use caller's chunk size, and the default multiple of it * for the maximum allocation */ max_allocation = CONN_MEM_STD_NUMBER_CHUNKS * chunk_size; } else if (chunk_size > max_allocation) { /* chunk size is larger than max allocation */ throw sk_error_conn_param_value_t( ("Specified chunk size greater than" " specified max allocation")); } /* else both values specified and are valid */ } else if (max_allocation) { if (max_allocation < CONN_MEM_MIN_CHUNK_SIZE) { std::ostringstream os; os << "Specified max allocation is below minimum of " << CONN_MEM_MIN_CHUNK_SIZE; throw sk_error_conn_param_value_t(os.str()); } /* determine the chunk size */ if (max_allocation >= CONN_MEM_STD_CHUNK_SIZE) { /* use standard chunk size */ chunk_size = CONN_MEM_STD_CHUNK_SIZE; } else { chunk_size = (max_allocation / CONN_MEM_STD_NUMBER_CHUNKS); if (chunk_size < CONN_MEM_MIN_CHUNK_SIZE) { chunk_size = CONN_MEM_MIN_CHUNK_SIZE; } } } else { /* use the default sizes */ chunk_size = CONN_MEM_STD_CHUNK_SIZE; max_allocation = CONN_MEM_STD_NUMBER_CHUNKS * chunk_size; } block_max_size = BLOCK_MAX_SIZE_FROM_CHUNK_SIZE(chunk_size); initialized = true; } /* linked list of chunks of memory */ conn_mem_chunk_t *writer_chunk; conn_mem_chunk_t *reader_chunk; conn_mem_chunk_t *spare_chunk; /* size of an individual chunk of memory; may be specified by the * caller */ size_t chunk_size; /* maximum block size a module may request; calculated as * roughly 1/3 of the chunk_size */ size_t block_max_size; enum param_id_t { CONN_MEM_CHUNK_SIZE, CONN_MEM_MAX_ALLOCATION }; struct class_params_t { const char *key; int val; const char *desciption; }; static MemoryConnector::class_params_t class_params[] = { {"chunk_size", CONN_MEM_CHUNK_SIZE, "Size of an individual chunk of memory"}, {"max_allocation", CONN_MEM_MAX_ALLOCATION, ("Maximum amount of memory that may be allocated across all blocks" " on all chunks; does not include internal overhead")}, {0, 0, 0} /* sentinel */ }; #if 0 static const std::set& create_param_defs() { size_t i; std::set defs; for (i = 0; class_params[i].key != NULL; ++i) { assert(i == (size_t)class_params[i].val); Parameterized::Definition d(class_params[i].key, class_params[i].desciption, (void*)&class_params[i].val, 1, 1); assert(defs.find(d) = defs.end()); defs.insert(d); } return defs; } #endif }; } /* ** Local Variables: ** mode:c++ ** indent-tabs-mode:nil ** c-basic-offset:4 ** End: */ --=-=-= Content-Type: text/plain -- Michael Welsh Duggan (mwd@cert.org) --=-=-=--