3 * Copyright (c) 2000 by Gilbert Ramirez <gram@alumni.rice.edu>
5 * Wireshark - Network traffic analyzer
6 * By Gerald Combs <gerald@wireshark.org>
7 * Copyright 1998 Gerald Combs
9 * This program is free software; you can redistribute it and/or
10 * modify it under the terms of the GNU General Public License
11 * as published by the Free Software Foundation; either version 2
12 * of the License, or (at your option) any later version.
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU General Public License for more details.
19 * You should have received a copy of the GNU General Public License
20 * along with this program; if not, write to the Free Software
21 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
26 #include <epan/emem.h>
29 #include "tvbuff-int.h"
30 #include "proto.h" /* XXX - only used for DISSECTOR_ASSERT, probably a new header file? */
31 #include "exceptions.h"
34 /** The backing tvbuff_t */
37 /** The offset of 'tvb' to which I'm privy */
39 /** The length of 'tvb' to which I'm privy */
51 subset_offset(const tvbuff_t *tvb, const guint counter)
53 const struct tvb_subset *subset_tvb = (const struct tvb_subset *) tvb;
54 const tvbuff_t *member = subset_tvb->subset.tvb;
56 return tvb_offset_from_real_beginning_counter(member, counter + subset_tvb->subset.offset);
60 subset_memcpy(tvbuff_t *tvb, void *target, guint abs_offset, guint abs_length)
62 struct tvb_subset *subset_tvb = (struct tvb_subset *) tvb;
64 return tvb_memcpy(subset_tvb->subset.tvb, target, subset_tvb->subset.offset + abs_offset, abs_length);
68 subset_get_ptr(tvbuff_t *tvb, guint abs_offset, guint abs_length)
70 struct tvb_subset *subset_tvb = (struct tvb_subset *) tvb;
72 return tvb_get_ptr(subset_tvb->subset.tvb, subset_tvb->subset.offset + abs_offset, abs_length);
76 subset_find_guint8(tvbuff_t *tvb, guint abs_offset, guint limit, guint8 needle)
78 struct tvb_subset *subset_tvb = (struct tvb_subset *) tvb;
80 return tvb_find_guint8(subset_tvb->subset.tvb, subset_tvb->subset.offset + abs_offset, limit, needle);
84 subset_pbrk_guint8(tvbuff_t *tvb, guint abs_offset, guint limit, const guint8 *needles, guchar *found_needle)
86 struct tvb_subset *subset_tvb = (struct tvb_subset *) tvb;
88 return tvb_pbrk_guint8(subset_tvb->subset.tvb, subset_tvb->subset.offset + abs_offset, limit, needles, found_needle);
92 subset_clone(tvbuff_t *tvb, guint abs_offset, guint abs_length)
94 struct tvb_subset *subset_tvb = (struct tvb_subset *) tvb;
96 return tvb_clone_offset_len(subset_tvb->subset.tvb, subset_tvb->subset.offset + abs_offset, abs_length);
99 static const struct tvb_ops tvb_subset_ops = {
100 sizeof(struct tvb_subset), /* size */
103 subset_offset, /* offset */
104 subset_get_ptr, /* get_ptr */
105 subset_memcpy, /* memcpy */
106 subset_find_guint8, /* find_guint8 */
107 subset_pbrk_guint8, /* pbrk_guint8 */
108 subset_clone, /* clone */
112 tvb_new_with_subset(tvbuff_t *backing, const gint reported_length,
113 const guint subset_tvb_offset, const guint subset_tvb_length)
115 tvbuff_t *tvb = tvb_new(&tvb_subset_ops);
116 struct tvb_subset *subset_tvb = (struct tvb_subset *) tvb;
118 subset_tvb->subset.offset = subset_tvb_offset;
119 subset_tvb->subset.length = subset_tvb_length;
121 subset_tvb->subset.tvb = backing;
122 tvb->length = subset_tvb_length;
123 tvb->flags = backing->flags;
125 if (reported_length == -1) {
126 tvb->reported_length = backing->reported_length - subset_tvb_offset;
129 tvb->reported_length = reported_length;
131 tvb->initialized = TRUE;
133 /* Optimization. If the backing buffer has a pointer to contiguous, real data,
134 * then we can point directly to our starting offset in that buffer */
135 if (backing->real_data != NULL) {
136 tvb->real_data = backing->real_data + subset_tvb_offset;
140 * The top-level data source of this tvbuff is the top-level
141 * data source of its parent.
143 tvb->ds_tvb = backing->ds_tvb;
149 tvb_new_subset(tvbuff_t *backing, const gint backing_offset, const gint backing_length, const gint reported_length)
152 guint subset_tvb_offset;
153 guint subset_tvb_length;
155 DISSECTOR_ASSERT(backing && backing->initialized);
157 THROW_ON(reported_length < -1, ReportedBoundsError);
159 tvb_check_offset_length(backing, backing_offset, backing_length,
163 tvb = tvb_new_with_subset(backing, reported_length,
164 subset_tvb_offset, subset_tvb_length);
166 tvb_add_to_chain(backing, tvb);
172 tvb_new_subset_length(tvbuff_t *backing, const gint backing_offset, const gint backing_length)
174 gint captured_length;
176 guint subset_tvb_offset;
177 guint subset_tvb_length;
179 DISSECTOR_ASSERT(backing && backing->initialized);
181 THROW_ON(backing_length < 0, ReportedBoundsError);
184 * Give the next dissector only captured_length bytes.
186 captured_length = tvb_length_remaining(backing, backing_offset);
187 THROW_ON(captured_length < 0, BoundsError);
188 if (captured_length > backing_length)
189 captured_length = backing_length;
191 tvb_check_offset_length(backing, backing_offset, captured_length,
195 tvb = tvb_new_with_subset(backing, backing_length,
196 subset_tvb_offset, subset_tvb_length);
198 tvb_add_to_chain(backing, tvb);
204 tvb_new_subset_remaining(tvbuff_t *backing, const gint backing_offset)
207 guint subset_tvb_offset;
208 guint subset_tvb_length;
210 tvb_check_offset_length(backing, backing_offset, -1 /* backing_length */,
214 tvb = tvb_new_with_subset(backing, -1 /* reported_length */,
215 subset_tvb_offset, subset_tvb_length);
217 tvb_add_to_chain(backing, tvb);
223 tvb_new_proxy(tvbuff_t *backing)
228 tvb = tvb_new_with_subset(backing, backing->reported_length, 0, backing->length);
230 tvb = tvb_new_real_data(NULL, 0, 0);
239 * Editor modelines - http://www.wireshark.org/tools/modelines.html
244 * indent-tabs-mode: t
247 * vi: set shiftwidth=8 tabstop=8 noexpandtab:
248 * :indentSize=8:tabSize=8:noTabs=false: