285 lines
		
	
	
		
			9.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
		
		
			
		
	
	
			285 lines
		
	
	
		
			9.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
|  | /*
 | ||
|  |  * libwebsockets - small server side websockets and web server implementation | ||
|  |  * | ||
|  |  * Copyright (C) 2010 - 2019 Andy Green <andy@warmcat.com> | ||
|  |  * | ||
|  |  * Permission is hereby granted, free of charge, to any person obtaining a copy | ||
|  |  * of this software and associated documentation files (the "Software"), to | ||
|  |  * deal in the Software without restriction, including without limitation the | ||
|  |  * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or | ||
|  |  * sell copies of the Software, and to permit persons to whom the Software is | ||
|  |  * furnished to do so, subject to the following conditions: | ||
|  |  * | ||
|  |  * The above copyright notice and this permission notice shall be included in | ||
|  |  * all copies or substantial portions of the Software. | ||
|  |  * | ||
|  |  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||
|  |  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | ||
|  |  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | ||
|  |  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | ||
|  |  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING | ||
|  |  * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS | ||
|  |  * IN THE SOFTWARE. | ||
|  |  */ | ||
|  | 
 | ||
|  | /*! \defgroup fops file operation wrapping
 | ||
|  |  * | ||
|  |  * ##File operation wrapping | ||
|  |  * | ||
|  |  * Use these helper functions if you want to access a file from the perspective | ||
|  |  * of a specific wsi, which is usually the case.  If you just want contextless | ||
|  |  * file access, use the fops callbacks directly with NULL wsi instead of these | ||
|  |  * helpers. | ||
|  |  * | ||
|  |  * If so, then it calls the platform handler or user overrides where present | ||
|  |  * (as defined in info->fops) | ||
|  |  * | ||
|  |  * The advantage from all this is user code can be portable for file operations | ||
|  |  * without having to deal with differences between platforms. | ||
|  |  */ | ||
|  | //@{
 | ||
|  | 
 | ||
|  | /** struct lws_plat_file_ops - Platform-specific file operations
 | ||
|  |  * | ||
|  |  * These provide platform-agnostic ways to deal with filesystem access in the | ||
|  |  * library and in the user code. | ||
|  |  */ | ||
|  | 
 | ||
|  | #if defined(LWS_PLAT_FREERTOS)
 | ||
|  | /* sdk preprocessor defs? compiler issue? gets confused with member names */ | ||
|  | #define LWS_FOP_OPEN		_open
 | ||
|  | #define LWS_FOP_CLOSE		_close
 | ||
|  | #define LWS_FOP_SEEK_CUR	_seek_cur
 | ||
|  | #define LWS_FOP_READ		_read
 | ||
|  | #define LWS_FOP_WRITE		_write
 | ||
|  | #else
 | ||
|  | #define LWS_FOP_OPEN		open
 | ||
|  | #define LWS_FOP_CLOSE		close
 | ||
|  | #define LWS_FOP_SEEK_CUR	seek_cur
 | ||
|  | #define LWS_FOP_READ		read
 | ||
|  | #define LWS_FOP_WRITE		write
 | ||
|  | #endif
 | ||
|  | 
 | ||
|  | #define LWS_FOP_FLAGS_MASK		   ((1 << 23) - 1)
 | ||
|  | #define LWS_FOP_FLAG_COMPR_ACCEPTABLE_GZIP (1 << 24)
 | ||
|  | #define LWS_FOP_FLAG_COMPR_IS_GZIP	   (1 << 25)
 | ||
|  | #define LWS_FOP_FLAG_MOD_TIME_VALID	   (1 << 26)
 | ||
|  | #define LWS_FOP_FLAG_VIRTUAL		   (1 << 27)
 | ||
|  | 
 | ||
|  | struct lws_plat_file_ops; | ||
|  | 
 | ||
|  | struct lws_fop_fd { | ||
|  | 	lws_filefd_type			fd; | ||
|  | 	/**< real file descriptor related to the file... */ | ||
|  | 	const struct lws_plat_file_ops	*fops; | ||
|  | 	/**< fops that apply to this fop_fd */ | ||
|  | 	void				*filesystem_priv; | ||
|  | 	/**< ignored by lws; owned by the fops handlers */ | ||
|  | 	lws_filepos_t			pos; | ||
|  | 	/**< generic "position in file" */ | ||
|  | 	lws_filepos_t			len; | ||
|  | 	/**< generic "length of file" */ | ||
|  | 	lws_fop_flags_t			flags; | ||
|  | 	/**< copy of the returned flags */ | ||
|  | 	uint32_t			mod_time; | ||
|  | 	/**< optional "modification time of file", only valid if .open()
 | ||
|  | 	 * set the LWS_FOP_FLAG_MOD_TIME_VALID flag */ | ||
|  | }; | ||
|  | typedef struct lws_fop_fd *lws_fop_fd_t; | ||
|  | 
 | ||
|  | struct lws_fops_index { | ||
|  | 	const char *sig;	/* NULL or vfs signature, eg, ".zip/" */ | ||
|  | 	uint8_t len;		/* length of above string */ | ||
|  | }; | ||
|  | 
 | ||
|  | struct lws_plat_file_ops { | ||
|  | 	lws_fop_fd_t (*LWS_FOP_OPEN)(const struct lws_plat_file_ops *fops_own, | ||
|  | 				     const struct lws_plat_file_ops *fops, | ||
|  | 				     const char *filename, const char *vpath, | ||
|  | 				     lws_fop_flags_t *flags); | ||
|  | 	/**< Open file (always binary access if plat supports it)
 | ||
|  | 	 * fops_own is the fops this was called through.  fops is the base | ||
|  | 	 * fops the open can use to find files to process as present as its own, | ||
|  | 	 * like the zip fops does. | ||
|  | 	 * | ||
|  | 	 * vpath may be NULL, or if the fops understands it, the point at which | ||
|  | 	 * the filename's virtual part starts. | ||
|  | 	 * *flags & LWS_FOP_FLAGS_MASK should be set to O_RDONLY or O_RDWR. | ||
|  | 	 * If the file may be gzip-compressed, | ||
|  | 	 * LWS_FOP_FLAG_COMPR_ACCEPTABLE_GZIP is set.  If it actually is | ||
|  | 	 * gzip-compressed, then the open handler should OR | ||
|  | 	 * LWS_FOP_FLAG_COMPR_IS_GZIP on to *flags before returning. | ||
|  | 	 */ | ||
|  | 	int (*LWS_FOP_CLOSE)(lws_fop_fd_t *fop_fd); | ||
|  | 	/**< close file AND set the pointer to NULL */ | ||
|  | 	lws_fileofs_t (*LWS_FOP_SEEK_CUR)(lws_fop_fd_t fop_fd, | ||
|  | 					  lws_fileofs_t offset_from_cur_pos); | ||
|  | 	/**< seek from current position */ | ||
|  | 	int (*LWS_FOP_READ)(lws_fop_fd_t fop_fd, lws_filepos_t *amount, | ||
|  | 			    uint8_t *buf, lws_filepos_t len); | ||
|  | 	/**< Read from file, on exit *amount is set to amount actually read */ | ||
|  | 	int (*LWS_FOP_WRITE)(lws_fop_fd_t fop_fd, lws_filepos_t *amount, | ||
|  | 			     uint8_t *buf, lws_filepos_t len); | ||
|  | 	/**< Write to file, on exit *amount is set to amount actually written */ | ||
|  | 
 | ||
|  | 	struct lws_fops_index fi[3]; | ||
|  | 	/**< vfs path signatures implying use of this fops */ | ||
|  | 
 | ||
|  | 	const struct lws_plat_file_ops *next; | ||
|  | 	/**< NULL or next fops in list... eg copy static fops def to heap
 | ||
|  | 	 * and modify copy at runtime */ | ||
|  | 
 | ||
|  | 	struct lws_context		*cx; | ||
|  | 	/**< the lws_context...  eg copy static fops def to heap
 | ||
|  | 	 * and modify copy at runtime */ | ||
|  | 
 | ||
|  | 	/* Add new things just above here ---^
 | ||
|  | 	 * This is part of the ABI, don't needlessly break compatibility */ | ||
|  | }; | ||
|  | 
 | ||
|  | /**
 | ||
|  |  * lws_get_fops() - get current file ops | ||
|  |  * | ||
|  |  * \param context: context | ||
|  |  */ | ||
|  | LWS_VISIBLE LWS_EXTERN struct lws_plat_file_ops * LWS_WARN_UNUSED_RESULT | ||
|  | lws_get_fops(struct lws_context *context); | ||
|  | LWS_VISIBLE LWS_EXTERN void | ||
|  | lws_set_fops(struct lws_context *context, const struct lws_plat_file_ops *fops); | ||
|  | /**
 | ||
|  |  * lws_vfs_tell() - get current file position | ||
|  |  * | ||
|  |  * \param fop_fd: fop_fd we are asking about | ||
|  |  */ | ||
|  | LWS_VISIBLE LWS_EXTERN lws_filepos_t LWS_WARN_UNUSED_RESULT | ||
|  | lws_vfs_tell(lws_fop_fd_t fop_fd); | ||
|  | /**
 | ||
|  |  * lws_vfs_get_length() - get current file total length in bytes | ||
|  |  * | ||
|  |  * \param fop_fd: fop_fd we are asking about | ||
|  |  */ | ||
|  | LWS_VISIBLE LWS_EXTERN lws_filepos_t LWS_WARN_UNUSED_RESULT | ||
|  | lws_vfs_get_length(lws_fop_fd_t fop_fd); | ||
|  | /**
 | ||
|  |  * lws_vfs_get_mod_time() - get time file last modified | ||
|  |  * | ||
|  |  * \param fop_fd: fop_fd we are asking about | ||
|  |  */ | ||
|  | LWS_VISIBLE LWS_EXTERN uint32_t LWS_WARN_UNUSED_RESULT | ||
|  | lws_vfs_get_mod_time(lws_fop_fd_t fop_fd); | ||
|  | /**
 | ||
|  |  * lws_vfs_file_seek_set() - seek relative to start of file | ||
|  |  * | ||
|  |  * \param fop_fd: fop_fd we are seeking in | ||
|  |  * \param offset: offset from start of file | ||
|  |  */ | ||
|  | LWS_VISIBLE LWS_EXTERN lws_fileofs_t | ||
|  | lws_vfs_file_seek_set(lws_fop_fd_t fop_fd, lws_fileofs_t offset); | ||
|  | /**
 | ||
|  |  * lws_vfs_file_seek_end() - seek relative to end of file | ||
|  |  * | ||
|  |  * \param fop_fd: fop_fd we are seeking in | ||
|  |  * \param offset: offset from start of file | ||
|  |  */ | ||
|  | LWS_VISIBLE LWS_EXTERN lws_fileofs_t | ||
|  | lws_vfs_file_seek_end(lws_fop_fd_t fop_fd, lws_fileofs_t offset); | ||
|  | 
 | ||
|  | extern struct lws_plat_file_ops fops_zip; | ||
|  | 
 | ||
|  | /**
 | ||
|  |  * lws_plat_file_open() - open vfs filepath | ||
|  |  * | ||
|  |  * \param fops: file ops struct that applies to this descriptor | ||
|  |  * \param vfs_path: filename to open | ||
|  |  * \param flags: pointer to open flags | ||
|  |  * | ||
|  |  * The vfs_path is scanned for known fops signatures, and the open directed | ||
|  |  * to any matching fops open. | ||
|  |  * | ||
|  |  * User code should use this api to perform vfs opens. | ||
|  |  * | ||
|  |  * returns semi-opaque handle | ||
|  |  */ | ||
|  | LWS_VISIBLE LWS_EXTERN lws_fop_fd_t LWS_WARN_UNUSED_RESULT | ||
|  | lws_vfs_file_open(const struct lws_plat_file_ops *fops, const char *vfs_path, | ||
|  | 		  lws_fop_flags_t *flags); | ||
|  | 
 | ||
|  | /**
 | ||
|  |  * lws_plat_file_close() - close file | ||
|  |  * | ||
|  |  * \param fop_fd: file handle to close | ||
|  |  */ | ||
|  | static LWS_INLINE int | ||
|  | lws_vfs_file_close(lws_fop_fd_t *fop_fd) | ||
|  | { | ||
|  | 	if (*fop_fd && (*fop_fd)->fops) | ||
|  | 		return (*fop_fd)->fops->LWS_FOP_CLOSE(fop_fd); | ||
|  | 
 | ||
|  | 	return 0; | ||
|  | } | ||
|  | 
 | ||
|  | /**
 | ||
|  |  * lws_plat_file_seek_cur() - close file | ||
|  |  * | ||
|  |  * | ||
|  |  * \param fop_fd: file handle | ||
|  |  * \param offset: position to seek to | ||
|  |  */ | ||
|  | static LWS_INLINE lws_fileofs_t | ||
|  | lws_vfs_file_seek_cur(lws_fop_fd_t fop_fd, lws_fileofs_t offset) | ||
|  | { | ||
|  | 	return fop_fd->fops->LWS_FOP_SEEK_CUR(fop_fd, offset); | ||
|  | } | ||
|  | /**
 | ||
|  |  * lws_plat_file_read() - read from file | ||
|  |  * | ||
|  |  * \param fop_fd: file handle | ||
|  |  * \param amount: how much to read (rewritten by call) | ||
|  |  * \param buf: buffer to write to | ||
|  |  * \param len: max length | ||
|  |  */ | ||
|  | static LWS_INLINE int LWS_WARN_UNUSED_RESULT | ||
|  | lws_vfs_file_read(lws_fop_fd_t fop_fd, lws_filepos_t *amount, | ||
|  | 		   uint8_t *buf, lws_filepos_t len) | ||
|  | { | ||
|  | 	return fop_fd->fops->LWS_FOP_READ(fop_fd, amount, buf, len); | ||
|  | } | ||
|  | /**
 | ||
|  |  * lws_plat_file_write() - write from file | ||
|  |  * | ||
|  |  * \param fop_fd: file handle | ||
|  |  * \param amount: how much to write (rewritten by call) | ||
|  |  * \param buf: buffer to read from | ||
|  |  * \param len: max length | ||
|  |  */ | ||
|  | static LWS_INLINE int LWS_WARN_UNUSED_RESULT | ||
|  | lws_vfs_file_write(lws_fop_fd_t fop_fd, lws_filepos_t *amount, | ||
|  | 		    uint8_t *buf, lws_filepos_t len) | ||
|  | { | ||
|  | 	return fop_fd->fops->LWS_FOP_WRITE(fop_fd, amount, buf, len); | ||
|  | } | ||
|  | 
 | ||
|  | /* these are the platform file operations implementations... they can
 | ||
|  |  * be called directly and used in fops arrays | ||
|  |  */ | ||
|  | 
 | ||
|  | LWS_VISIBLE LWS_EXTERN lws_fop_fd_t | ||
|  | _lws_plat_file_open(const struct lws_plat_file_ops *fops_own, | ||
|  | 		    const struct lws_plat_file_ops *fops, const char *filename, | ||
|  | 		    const char *vpath, lws_fop_flags_t *flags); | ||
|  | LWS_VISIBLE LWS_EXTERN int | ||
|  | _lws_plat_file_close(lws_fop_fd_t *fop_fd); | ||
|  | LWS_VISIBLE LWS_EXTERN lws_fileofs_t | ||
|  | _lws_plat_file_seek_cur(lws_fop_fd_t fop_fd, lws_fileofs_t offset); | ||
|  | LWS_VISIBLE LWS_EXTERN int | ||
|  | _lws_plat_file_read(lws_fop_fd_t fop_fd, lws_filepos_t *amount, | ||
|  | 		    uint8_t *buf, lws_filepos_t len); | ||
|  | LWS_VISIBLE LWS_EXTERN int | ||
|  | _lws_plat_file_write(lws_fop_fd_t fop_fd, lws_filepos_t *amount, | ||
|  | 		     uint8_t *buf, lws_filepos_t len); | ||
|  | 
 | ||
|  | LWS_VISIBLE LWS_EXTERN int | ||
|  | lws_alloc_vfs_file(struct lws_context *context, const char *filename, | ||
|  | 		   uint8_t **buf, lws_filepos_t *amount); | ||
|  | //@}
 |