Branch data Line data Source code
1 : : /*
2 : : * This file is part of the MicroPython project, http://micropython.org/
3 : : *
4 : : * The MIT License (MIT)
5 : : *
6 : : * Copyright (c) 2013-2016 Damien P. George
7 : : *
8 : : * Permission is hereby granted, free of charge, to any person obtaining a copy
9 : : * of this software and associated documentation files (the "Software"), to deal
10 : : * in the Software without restriction, including without limitation the rights
11 : : * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
12 : : * copies of the Software, and to permit persons to whom the Software is
13 : : * furnished to do so, subject to the following conditions:
14 : : *
15 : : * The above copyright notice and this permission notice shall be included in
16 : : * all copies or substantial portions of the Software.
17 : : *
18 : : * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
19 : : * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
20 : : * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
21 : : * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
22 : : * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
23 : : * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
24 : : * THE SOFTWARE.
25 : : */
26 : :
27 : : #include <stdio.h>
28 : : #include <assert.h>
29 : :
30 : : #include "py/runtime.h"
31 : : #include "py/mperrno.h"
32 : : #include "py/mpthread.h"
33 : : #include "py/reader.h"
34 : :
35 : : typedef struct _mp_reader_mem_t {
36 : : size_t free_len; // if >0 mem is freed on close by: m_free(beg, free_len)
37 : : const byte *beg;
38 : : const byte *cur;
39 : : const byte *end;
40 : : } mp_reader_mem_t;
41 : :
42 : 168817 : static mp_uint_t mp_reader_mem_readbyte(void *data) {
43 : 168817 : mp_reader_mem_t *reader = (mp_reader_mem_t *)data;
44 [ + + ]: 168817 : if (reader->cur < reader->end) {
45 : 162192 : return *reader->cur++;
46 : : } else {
47 : : return MP_READER_EOF;
48 : : }
49 : : }
50 : :
51 : 1726 : static void mp_reader_mem_close(void *data) {
52 : 1726 : mp_reader_mem_t *reader = (mp_reader_mem_t *)data;
53 [ - + ]: 1726 : if (reader->free_len > 0 && reader->free_len != MP_READER_IS_ROM) {
54 : 0 : m_del(char, (char *)reader->beg, reader->free_len);
55 : : }
56 : 1726 : m_del_obj(mp_reader_mem_t, reader);
57 : 1726 : }
58 : :
59 : 1732 : void mp_reader_new_mem(mp_reader_t *reader, const byte *buf, size_t len, size_t free_len) {
60 : 1732 : mp_reader_mem_t *rm = m_new_obj(mp_reader_mem_t);
61 : 1732 : rm->free_len = free_len;
62 : 1732 : rm->beg = buf;
63 : 1732 : rm->cur = buf;
64 : 1732 : rm->end = buf + len;
65 : 1732 : reader->data = rm;
66 : 1732 : reader->readbyte = mp_reader_mem_readbyte;
67 : 1732 : reader->close = mp_reader_mem_close;
68 : 1732 : }
69 : :
70 : 18549 : const uint8_t *mp_reader_try_read_rom(mp_reader_t *reader, size_t len) {
71 [ + + ]: 18549 : if (reader->readbyte != mp_reader_mem_readbyte) {
72 : : return NULL;
73 : : }
74 : 22 : mp_reader_mem_t *m = reader->data;
75 [ + - ]: 22 : if (m->free_len != MP_READER_IS_ROM) {
76 : : return NULL;
77 : : }
78 : 22 : const uint8_t *data = m->cur;
79 : 22 : m->cur += len;
80 : 22 : return data;
81 : : }
82 : :
83 : : #if MICROPY_READER_POSIX
84 : :
85 : : #include <sys/stat.h>
86 : : #include <fcntl.h>
87 : : #include <unistd.h>
88 : :
89 : : typedef struct _mp_reader_posix_t {
90 : : bool close_fd;
91 : : int fd;
92 : : size_t len;
93 : : size_t pos;
94 : : byte buf[20];
95 : : } mp_reader_posix_t;
96 : :
97 : 81 : static mp_uint_t mp_reader_posix_readbyte(void *data) {
98 : 81 : mp_reader_posix_t *reader = (mp_reader_posix_t *)data;
99 [ + + ]: 81 : if (reader->pos >= reader->len) {
100 [ + + ]: 6 : if (reader->len == 0) {
101 : : return MP_READER_EOF;
102 : : } else {
103 : 4 : MP_THREAD_GIL_EXIT();
104 : 4 : int n = read(reader->fd, reader->buf, sizeof(reader->buf));
105 : 4 : MP_THREAD_GIL_ENTER();
106 [ + + ]: 4 : if (n <= 0) {
107 : 1 : reader->len = 0;
108 : 1 : return MP_READER_EOF;
109 : : }
110 : 3 : reader->len = n;
111 : 3 : reader->pos = 0;
112 : : }
113 : : }
114 : 78 : return reader->buf[reader->pos++];
115 : : }
116 : :
117 : 1 : static void mp_reader_posix_close(void *data) {
118 : 1 : mp_reader_posix_t *reader = (mp_reader_posix_t *)data;
119 [ - + ]: 1 : if (reader->close_fd) {
120 : 0 : MP_THREAD_GIL_EXIT();
121 : 0 : close(reader->fd);
122 : 1 : MP_THREAD_GIL_ENTER();
123 : : }
124 : 1 : m_del_obj(mp_reader_posix_t, reader);
125 : 1 : }
126 : :
127 : 1 : void mp_reader_new_file_from_fd(mp_reader_t *reader, int fd, bool close_fd) {
128 : 1 : mp_reader_posix_t *rp = m_new_obj(mp_reader_posix_t);
129 : 1 : rp->close_fd = close_fd;
130 : 1 : rp->fd = fd;
131 : 1 : MP_THREAD_GIL_EXIT();
132 : 1 : int n = read(rp->fd, rp->buf, sizeof(rp->buf));
133 [ - + ]: 1 : if (n == -1) {
134 [ # # ]: 0 : if (close_fd) {
135 : 0 : close(fd);
136 : : }
137 : 0 : MP_THREAD_GIL_ENTER();
138 : 0 : mp_raise_OSError(errno);
139 : : }
140 : 1 : MP_THREAD_GIL_ENTER();
141 : 1 : rp->len = n;
142 : 1 : rp->pos = 0;
143 : 1 : reader->data = rp;
144 : 1 : reader->readbyte = mp_reader_posix_readbyte;
145 : 1 : reader->close = mp_reader_posix_close;
146 : 1 : }
147 : :
148 : : #if !MICROPY_VFS_POSIX
149 : : // If MICROPY_VFS_POSIX is defined then this function is provided by the VFS layer
150 : : void mp_reader_new_file(mp_reader_t *reader, qstr filename) {
151 : : MP_THREAD_GIL_EXIT();
152 : : int fd = open(qstr_str(filename), O_RDONLY, 0644);
153 : : MP_THREAD_GIL_ENTER();
154 : : if (fd < 0) {
155 : : mp_raise_OSError_with_filename(errno, qstr_str(filename));
156 : : }
157 : : mp_reader_new_file_from_fd(reader, fd, true);
158 : : }
159 : : #endif
160 : :
161 : : #endif
|