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 
mp_reader_mem_readbyte(void * data)42 STATIC mp_uint_t mp_reader_mem_readbyte(void *data) {
43     mp_reader_mem_t *reader = (mp_reader_mem_t *)data;
44     if (reader->cur < reader->end) {
45         return *reader->cur++;
46     } else {
47         return MP_READER_EOF;
48     }
49 }
50 
mp_reader_mem_close(void * data)51 STATIC void mp_reader_mem_close(void *data) {
52     mp_reader_mem_t *reader = (mp_reader_mem_t *)data;
53     if (reader->free_len > 0) {
54         m_del(char, (char *)reader->beg, reader->free_len);
55     }
56     m_del_obj(mp_reader_mem_t, reader);
57 }
58 
mp_reader_new_mem(mp_reader_t * reader,const byte * buf,size_t len,size_t free_len)59 void mp_reader_new_mem(mp_reader_t *reader, const byte *buf, size_t len, size_t free_len) {
60     mp_reader_mem_t *rm = m_new_obj(mp_reader_mem_t);
61     rm->free_len = free_len;
62     rm->beg = buf;
63     rm->cur = buf;
64     rm->end = buf + len;
65     reader->data = rm;
66     reader->readbyte = mp_reader_mem_readbyte;
67     reader->close = mp_reader_mem_close;
68 }
69 
70 #if MICROPY_READER_POSIX
71 
72 #include <sys/stat.h>
73 #include <fcntl.h>
74 #include <unistd.h>
75 
76 typedef struct _mp_reader_posix_t {
77     bool close_fd;
78     int fd;
79     size_t len;
80     size_t pos;
81     byte buf[20];
82 } mp_reader_posix_t;
83 
mp_reader_posix_readbyte(void * data)84 STATIC mp_uint_t mp_reader_posix_readbyte(void *data) {
85     mp_reader_posix_t *reader = (mp_reader_posix_t *)data;
86     if (reader->pos >= reader->len) {
87         if (reader->len == 0) {
88             return MP_READER_EOF;
89         } else {
90             MP_THREAD_GIL_EXIT();
91             int n = read(reader->fd, reader->buf, sizeof(reader->buf));
92             MP_THREAD_GIL_ENTER();
93             if (n <= 0) {
94                 reader->len = 0;
95                 return MP_READER_EOF;
96             }
97             reader->len = n;
98             reader->pos = 0;
99         }
100     }
101     return reader->buf[reader->pos++];
102 }
103 
mp_reader_posix_close(void * data)104 STATIC void mp_reader_posix_close(void *data) {
105     mp_reader_posix_t *reader = (mp_reader_posix_t *)data;
106     if (reader->close_fd) {
107         MP_THREAD_GIL_EXIT();
108         close(reader->fd);
109         MP_THREAD_GIL_ENTER();
110     }
111     m_del_obj(mp_reader_posix_t, reader);
112 }
113 
mp_reader_new_file_from_fd(mp_reader_t * reader,int fd,bool close_fd)114 void mp_reader_new_file_from_fd(mp_reader_t *reader, int fd, bool close_fd) {
115     mp_reader_posix_t *rp = m_new_obj(mp_reader_posix_t);
116     rp->close_fd = close_fd;
117     rp->fd = fd;
118     MP_THREAD_GIL_EXIT();
119     int n = read(rp->fd, rp->buf, sizeof(rp->buf));
120     if (n == -1) {
121         if (close_fd) {
122             close(fd);
123         }
124         MP_THREAD_GIL_ENTER();
125         mp_raise_OSError(errno);
126     }
127     MP_THREAD_GIL_ENTER();
128     rp->len = n;
129     rp->pos = 0;
130     reader->data = rp;
131     reader->readbyte = mp_reader_posix_readbyte;
132     reader->close = mp_reader_posix_close;
133 }
134 
135 // Modified bt HaaS begin
136 
137 // As HaaS set MICROPY_READER_POSIX and unset MICROPY_READER_VFS,
138 // no module supply mp_reader_new_file function.
139 // We remove '!' here to make sure mp_reader_new_file defined
140 
141 // #if !MICROPY_VFS_POSIX
142 #if MICROPY_VFS_POSIX
143 // Modified bt HaaS end
144 
145 // If MICROPY_VFS_POSIX is defined then this function is provided by the VFS layer
mp_reader_new_file(mp_reader_t * reader,const char * filename)146 void mp_reader_new_file(mp_reader_t *reader, const char *filename) {
147     MP_THREAD_GIL_EXIT();
148     int fd = open(filename, O_RDONLY, 0644);
149     MP_THREAD_GIL_ENTER();
150     if (fd < 0) {
151         mp_raise_OSError(errno);
152     }
153     mp_reader_new_file_from_fd(reader, fd, true);
154 }
155 #endif
156 
157 #endif
158