// 2 or 3 structures, select one that is before target point, closest to target /* This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see . */ /* ArduPilot filesystem interface for parameters */ #include "AP_Filesystem.h" #include "AP_Filesystem_Param.h" #include #include #define PACKED_NAME "param.pck" extern const AP_HAL::HAL& hal; extern int errno; int AP_Filesystem_Param::open(const char *fname, int flags) { if ((flags & O_ACCMODE) != O_RDONLY) { return -1; } uint8_t idx; for (idx=0; idx= max_open_file || !file[fd].open) { errno = EBADF; return -1; } struct rfile &r = file[fd]; r.open = false; delete [] r.cursors; return 0; } /* packed format: uint8_t type:4; // AP_Param type NONE=0, INT8=1, INT16=2, INT32=3, FLOAT=4 uint8_t type_len:4; // number of bytes in type uint8_t common_len:4; // number of name bytes in common with previous entry, 0..15 uint8_t name_len:4; // non-common length of param name -1 (0..15) uint8_t name[name_len]; // name uint8_t data[]; // value, may be truncated by record_length */ /* pack a single parameter */ uint8_t AP_Filesystem_Param::pack_param(const AP_Param *ap, const char *pname, const char *last_name, enum ap_var_type ptype, uint8_t *buf, uint8_t buflen) { uint8_t common_len = 0; while (*pname == *last_name) { common_len++; pname++; last_name++; } const uint8_t name_len = strlen(pname); uint8_t type_len = AP_Param::type_size(ptype); const uint8_t *pbuf = (const uint8_t *)ap; while (type_len > 0 && pbuf[type_len-1] == 0) { type_len--; } uint8_t packed_len = type_len + name_len + 2; if (packed_len <= buflen && buf) { buf[0] = uint8_t(ptype) | (type_len<<4); buf[1] = common_len | ((name_len-1)<<4); memcpy(&buf[2], pname, name_len); memcpy(&buf[2+name_len], ap, type_len); } return packed_len; } /* seek the token to match file offset */ bool AP_Filesystem_Param::token_seek(const struct rfile &r, struct cursor &c) { //hal.console->printf("token_seek %u %u\n", unsigned(r.file_ofs), unsigned(c.token_ofs)); if (r.file_ofs == 0) { memset(&c, 0, sizeof(c)); return true; } if (c.token_ofs > r.file_ofs) { memset(&c, 0, sizeof(c)); } char name[AP_MAX_NAME_SIZE+1]; name[AP_MAX_NAME_SIZE] = 0; enum ap_var_type ptype; if (c.trailer_len > 0) { uint8_t n = MIN(c.trailer_len, r.file_ofs - c.token_ofs); if (n != c.trailer_len) { memmove(&c.trailer[0], &c.trailer[n], c.trailer_len - n); } c.trailer_len -= n; c.token_ofs += n; } while (r.file_ofs != c.token_ofs) { AP_Param *ap; if (c.token_ofs == 0) { ap = AP_Param::first(&c.token, &ptype); } else { ap = AP_Param::next_scalar(&c.token, &ptype); } if (ap == nullptr) { break; } ap->copy_name_token(c.token, name, AP_MAX_NAME_SIZE, true); uint32_t available = r.file_ofs - c.token_ofs; uint8_t tbuf[max_pack_len]; uint8_t len = pack_param(ap, name, c.last_name, ptype, tbuf, max_pack_len); uint8_t n = MIN(len, available); if (len > available) { c.trailer_len = len - available; memcpy(c.trailer, &tbuf[n], c.trailer_len); } strcpy(c.last_name, name); c.token_ofs += n; } return r.file_ofs == c.token_ofs; } int32_t AP_Filesystem_Param::read(int fd, void *buf, uint32_t count) { if (fd < 0 || fd >= max_open_file || !file[fd].open) { errno = EBADF; return -1; } struct rfile &r = file[fd]; uint8_t best_i = 0; uint32_t best_ofs = r.cursors[0].token_ofs; // find the first cursor that is positioned after the file offset for (uint8_t i=1; i= r.file_ofs && c.token_ofs < best_ofs) { best_i = i; best_ofs = c.token_ofs; } } struct cursor &c = r.cursors[best_i]; if (r.file_ofs != c.token_ofs) { if (!token_seek(r, c)) { // must be EOF return 0; } } if (count == 0) { return 0; } uint8_t *ubuf = (uint8_t *)buf; char name[AP_MAX_NAME_SIZE+1]; name[AP_MAX_NAME_SIZE] = 0; size_t total = 0; if (c.trailer_len > 0) { uint8_t n = MIN(c.trailer_len, count); memcpy(ubuf, c.trailer, n); count -= n; ubuf += n; if (n != c.trailer_len) { memmove(&c.trailer[0], &c.trailer[n], c.trailer_len - n); } c.trailer_len -= n; total += n; c.token_ofs += n; } while (count > 0) { AP_Param *ap; enum ap_var_type ptype; if (c.token_ofs == 0) { ap = AP_Param::first(&c.token, &ptype); } else { ap = AP_Param::next_scalar(&c.token, &ptype); } if (ap == nullptr) { break; } ap->copy_name_token(c.token, name, AP_MAX_NAME_SIZE, true); uint8_t tbuf[max_pack_len]; uint8_t len = pack_param(ap, name, c.last_name, ptype, tbuf, sizeof(tbuf)); uint8_t n = MIN(len, count); if (len > count) { c.trailer_len = len - count; memcpy(c.trailer, &tbuf[count], c.trailer_len); } memcpy(ubuf, tbuf, n); strcpy(c.last_name, name); count -= n; ubuf += n; total += n; c.token_ofs += n; } r.file_ofs += total; return total; } int32_t AP_Filesystem_Param::lseek(int fd, int32_t offset, int seek_from) { if (fd < 0 || fd >= max_open_file || !file[fd].open) { errno = EBADF; return -1; } struct rfile &r = file[fd]; switch (seek_from) { case SEEK_SET: r.file_ofs = offset; break; case SEEK_CUR: r.file_ofs += offset; break; case SEEK_END: errno = EINVAL; return -1; } return r.file_ofs; } int AP_Filesystem_Param::stat(const char *name, struct stat *stbuf) { if (strcmp(name, PACKED_NAME) != 0) { errno = ENOENT; return -1; } memset(stbuf, 0, sizeof(*stbuf)); // give fixed size stbuf->st_size = 65535; return 0; }