blob: 5df45c4957a5960480c1af7e511a09a68b216000 [file] [log] [blame]
Damien George6b239c22016-11-16 16:04:57 +11001/*
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
Damien George18310342017-03-14 11:16:31 +110030#include "py/runtime.h"
Damien George6b239c22016-11-16 16:04:57 +110031#include "py/mperrno.h"
32#include "py/reader.h"
33
34typedef struct _mp_reader_mem_t {
35 size_t free_len; // if >0 mem is freed on close by: m_free(beg, free_len)
36 const byte *beg;
37 const byte *cur;
38 const byte *end;
39} mp_reader_mem_t;
40
41STATIC mp_uint_t mp_reader_mem_readbyte(void *data) {
42 mp_reader_mem_t *reader = (mp_reader_mem_t*)data;
43 if (reader->cur < reader->end) {
44 return *reader->cur++;
45 } else {
46 return MP_READER_EOF;
47 }
48}
49
50STATIC void mp_reader_mem_close(void *data) {
51 mp_reader_mem_t *reader = (mp_reader_mem_t*)data;
52 if (reader->free_len > 0) {
53 m_del(char, (char*)reader->beg, reader->free_len);
54 }
55 m_del_obj(mp_reader_mem_t, reader);
56}
57
Damien George18310342017-03-14 11:16:31 +110058void mp_reader_new_mem(mp_reader_t *reader, const byte *buf, size_t len, size_t free_len) {
59 mp_reader_mem_t *rm = m_new_obj(mp_reader_mem_t);
Damien George6b239c22016-11-16 16:04:57 +110060 rm->free_len = free_len;
61 rm->beg = buf;
62 rm->cur = buf;
63 rm->end = buf + len;
64 reader->data = rm;
65 reader->readbyte = mp_reader_mem_readbyte;
66 reader->close = mp_reader_mem_close;
Damien George6b239c22016-11-16 16:04:57 +110067}
68
69#if MICROPY_READER_POSIX
70
71#include <sys/stat.h>
72#include <fcntl.h>
73#include <unistd.h>
74#include <errno.h>
75
76typedef 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
84STATIC 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 int n = read(reader->fd, reader->buf, sizeof(reader->buf));
91 if (n <= 0) {
92 reader->len = 0;
93 return MP_READER_EOF;
94 }
95 reader->len = n;
96 reader->pos = 0;
97 }
98 }
99 return reader->buf[reader->pos++];
100}
101
102STATIC void mp_reader_posix_close(void *data) {
103 mp_reader_posix_t *reader = (mp_reader_posix_t*)data;
104 if (reader->close_fd) {
105 close(reader->fd);
106 }
107 m_del_obj(mp_reader_posix_t, reader);
108}
109
Damien George18310342017-03-14 11:16:31 +1100110void mp_reader_new_file_from_fd(mp_reader_t *reader, int fd, bool close_fd) {
111 mp_reader_posix_t *rp = m_new_obj(mp_reader_posix_t);
Damien George6b239c22016-11-16 16:04:57 +1100112 rp->close_fd = close_fd;
113 rp->fd = fd;
114 int n = read(rp->fd, rp->buf, sizeof(rp->buf));
115 if (n == -1) {
116 if (close_fd) {
117 close(fd);
118 }
Damien George18310342017-03-14 11:16:31 +1100119 mp_raise_OSError(errno);
Damien George6b239c22016-11-16 16:04:57 +1100120 }
121 rp->len = n;
122 rp->pos = 0;
123 reader->data = rp;
124 reader->readbyte = mp_reader_posix_readbyte;
125 reader->close = mp_reader_posix_close;
Damien George6b239c22016-11-16 16:04:57 +1100126}
127
Damien George18310342017-03-14 11:16:31 +1100128void mp_reader_new_file(mp_reader_t *reader, const char *filename) {
Damien George6b239c22016-11-16 16:04:57 +1100129 int fd = open(filename, O_RDONLY, 0644);
130 if (fd < 0) {
Damien George18310342017-03-14 11:16:31 +1100131 mp_raise_OSError(errno);
Damien George6b239c22016-11-16 16:04:57 +1100132 }
Damien George18310342017-03-14 11:16:31 +1100133 mp_reader_new_file_from_fd(reader, fd, true);
Damien George6b239c22016-11-16 16:04:57 +1100134}
135
136#endif