Split CTF IR and CTF writer APIs and implementations
[babeltrace.git] / include / babeltrace / ctf-writer / serialize-internal.h
1 #ifndef BABELTRACE_CTF_WRITER_SERIALIZE_INTERNAL_H
2 #define BABELTRACE_CTF_WRITER_SERIALIZE_INTERNAL_H
3
4 /*
5 * Copyright 2017 Philippe Proulx <pproulx@efficios.com>
6 *
7 * Permission is hereby granted, free of charge, to any person obtaining a copy
8 * of this software and associated documentation files (the "Software"), to deal
9 * in the Software without restriction, including without limitation the rights
10 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
11 * copies of the Software, and to permit persons to whom the Software is
12 * furnished to do so, subject to the following conditions:
13 *
14 * The above copyright notice and this permission notice shall be included in
15 * all copies or substantial portions of the Software.
16 *
17 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
18 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
19 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
20 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
21 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
22 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
23 * SOFTWARE.
24 *
25 * The Common Trace Format (CTF) Specification is available at
26 * http://www.efficios.com/ctf
27 */
28
29 #include <stdlib.h>
30 #include <stdint.h>
31 #include <limits.h>
32 #include <babeltrace/compat/mman-internal.h>
33 #include <sys/types.h>
34 #include <sys/stat.h>
35 #include <fcntl.h>
36 #include <babeltrace/align-internal.h>
37 #include <babeltrace/common-internal.h>
38 #include <babeltrace/mmap-align-internal.h>
39 #include <babeltrace/types.h>
40 #include <babeltrace/ctf-writer/fields.h>
41 #include <babeltrace/ctf-writer/field-types.h>
42 #include <babeltrace/ctf-ir/fields-internal.h>
43 #include <babeltrace/assert-internal.h>
44
45 #define PACKET_LEN_INCREMENT (bt_common_get_page_size() * 8 * CHAR_BIT)
46
47 struct bt_ctf_stream_pos {
48 int fd;
49 int prot; /* mmap protection */
50 int flags; /* mmap flags */
51
52 /* Current position */
53 off_t mmap_offset; /* mmap offset in the file, in bytes */
54 off_t mmap_base_offset; /* offset of start of packet in mmap, in bytes */
55 uint64_t packet_size; /* current packet size, in bits */
56 int64_t offset; /* offset from base, in bits. EOF for end of file. */
57 struct mmap_align *base_mma;/* mmap base address */
58 };
59
60 BT_HIDDEN
61 int bt_ctf_field_integer_write(struct bt_field_common *field,
62 struct bt_ctf_stream_pos *pos,
63 enum bt_ctf_byte_order native_byte_order);
64
65 BT_HIDDEN
66 int bt_ctf_field_floating_point_write(struct bt_field_common *field,
67 struct bt_ctf_stream_pos *pos,
68 enum bt_ctf_byte_order native_byte_order);
69
70 static inline
71 int bt_ctf_stream_pos_access_ok(struct bt_ctf_stream_pos *pos, uint64_t bit_len)
72 {
73 uint64_t max_len;
74
75 if (unlikely(pos->offset == EOF))
76 return 0;
77
78 if (pos->prot == PROT_READ) {
79 /*
80 * Reads may only reach up to the "content_size",
81 * regardless of the packet_size.
82 */
83 max_len = pos->offset;
84 } else {
85 /* Writes may take place up to the end of the packet. */
86 max_len = pos->packet_size;
87 }
88 if (unlikely(pos->offset < 0 || bit_len > INT64_MAX - pos->offset)) {
89 return 0;
90 }
91 if (unlikely(pos->offset + bit_len > max_len))
92 return 0;
93 return 1;
94 }
95
96 static inline
97 int bt_ctf_stream_pos_move(struct bt_ctf_stream_pos *pos, uint64_t bit_offset)
98 {
99 int ret = 0;
100
101 ret = bt_ctf_stream_pos_access_ok(pos, bit_offset);
102 if (!ret) {
103 goto end;
104 }
105 pos->offset += bit_offset;
106 end:
107 return ret;
108 }
109
110 static inline
111 int bt_ctf_stream_pos_align(struct bt_ctf_stream_pos *pos, uint64_t bit_offset)
112 {
113 return bt_ctf_stream_pos_move(pos,
114 offset_align(pos->offset, bit_offset));
115 }
116
117 static inline
118 char *bt_ctf_stream_pos_get_addr(struct bt_ctf_stream_pos *pos)
119 {
120 /* Only makes sense to get the address after aligning on CHAR_BIT */
121 BT_ASSERT(!(pos->offset % CHAR_BIT));
122 return ((char *) mmap_align_addr(pos->base_mma)) +
123 pos->mmap_base_offset + (pos->offset / CHAR_BIT);
124 }
125
126 static inline
127 int bt_ctf_stream_pos_init(struct bt_ctf_stream_pos *pos,
128 int fd, int open_flags)
129 {
130 pos->fd = fd;
131
132 switch (open_flags & O_ACCMODE) {
133 case O_RDONLY:
134 pos->prot = PROT_READ;
135 pos->flags = MAP_PRIVATE;
136 break;
137 case O_RDWR:
138 pos->prot = PROT_READ | PROT_WRITE;
139 pos->flags = MAP_SHARED;
140 break;
141 default:
142 abort();
143 }
144
145 return 0;
146 }
147
148 static inline
149 int bt_ctf_stream_pos_fini(struct bt_ctf_stream_pos *pos)
150 {
151 if (pos->base_mma) {
152 int ret;
153
154 /* unmap old base */
155 ret = munmap_align(pos->base_mma);
156 if (ret) {
157 return -1;
158 }
159 }
160
161 return 0;
162 }
163
164 BT_HIDDEN
165 void bt_ctf_stream_pos_packet_seek(struct bt_ctf_stream_pos *pos, size_t index,
166 int whence);
167
168 #endif /* BABELTRACE_CTF_WRITER_SERIALIZE_INTERNAL_H */
This page took 0.035795 seconds and 4 git commands to generate.