diff --git a/graphics/traveler/.gitignore b/graphics/traveler/.gitignore index 83bd7b811..c7cf4c8ae 100644 --- a/graphics/traveler/.gitignore +++ b/graphics/traveler/.gitignore @@ -1,6 +1,8 @@ /Make.dep /.depend /.built +/trv_romfs.h +/trv_romfs.img /*.asm /*.rel /*.lst diff --git a/graphics/traveler/Kconfig b/graphics/traveler/Kconfig index 47dd09c1b..719b5c3a1 100644 --- a/graphics/traveler/Kconfig +++ b/graphics/traveler/Kconfig @@ -4,6 +4,7 @@ # if GRAPHICS_TRAVELER +comment "Color configuration" choice prompt "Color format" @@ -17,6 +18,16 @@ config GRAPHICS_TRAVELER_RGB32_888 endchoice # Color format +config GRAPHICS_TRAVELER_PALRANGES + bool "Use ranged palette" + default y + ---help--- + If this option is false, then the whole palette table is loaded from + the graphics file. Otherwise, the palette table will be calculated + from a range table. Default y, this is a good thing. + +comment "Input device selection" + config GRAPHICS_TRAVELER_JOYSTICK bool default n @@ -49,10 +60,27 @@ config GRAPHICS_TRAVELER_JOYDEV default "/dev/djoy0" if GRAPHICS_TRAVELER_DJOYSTICK depends on GRAPHICS_TRAVELER_JOYSTICK +comment "Debug options" + +config GRAPHICS_TRAVELER_ROMFSDEMO + bool "Build ROMFS demo world" + default y + depends on FS_ROMFS + ---help--- + The traveller code will do nothing unless you also provide a world + data set for it to work on. The world/ sub-directory contains a + demo world suitable for testing. If this option is selected, then + the demo world is build as a mountable ROMFS file system. This new + function is then exported that can be used to mount the demo world: + + int trv_mount_world(int minor, FAR const char *mountpoint) + + This function must be called somewhere in your board-specific start- + up logic in order to mount the world at 'mountpoint' + config GRAPHICS_TRAVELER_PERFMON bool "Performance monitor" default y - depends on FS_READABLE ---help--- Enable or disable performance monitoring instrumentation and output. diff --git a/graphics/traveler/Makefile b/graphics/traveler/Makefile index 8008fb487..1f39d1af6 100644 --- a/graphics/traveler/Makefile +++ b/graphics/traveler/Makefile @@ -37,22 +37,39 @@ -include $(TOPDIR)/Make.defs include $(APPDIR)/Make.defs -# Hello, World! built-in application info +# Directories -APPNAME = traveler -PRIORITY = SCHED_PRIORITY_DEFAULT +TRAVELER_DIR = $(APPDIR)/graphics/traveler +TRAVELER_SRC = $(TRAVELER_DIR)/src +TRAVELER_INC = $(TRAVELER_DIR)/include +TRAVELER_DEMO = $(TRAVELER_DIR)/world + +# Traveler built-in application info + +APPNAME = traveler +PRIORITY = SCHED_PRIORITY_DEFAULT STACKSIZE = 2048 -# Hello, World! Example +# Traveler files + +ASRCS = +CSRCS = trv_bitmapfile.c trv_bitmaps.c trv_color.c trv_doors.c +CSRCS += trv_fsutils.c trv_graphicfile.c trv_graphics.c trv_input.c +CSRCS += trv_mem.c trv_paltbl.c trv_pcx.c trv_planefiles.c trv_planelists.c +CSRCS += trv_pov.c trv_rayavoid.c trv_raycast.c trv_raycntl.c +CSRCS += trv_rayprune.c trv_rayrend.c trv_texturefile.c trv_trigtbl.c +CSRCS += trv_world.c + +ifeq ($(CONFIG_GRAPHICS_TRAVELER_ROMFSDEMO),y) +CSRCS += trv_romfs.c +endif -ASRCS = -CSRCS = trv_color.c trv_graphics.c trv_input.c trv_mem.c trv_pov.c MAINSRC = trv_main.c ifeq ($(CONFIG_NX),y) -CSRCS += trv_nxbkgd.c +CSRCS += trv_nxbkgd.c ifeq ($(CONFIG_NX),y) -CSRCS += trv_nxserver.c +CSRCS += trv_nxserver.c endif endif @@ -67,6 +84,11 @@ ifneq ($(CONFIG_BUILD_KERNEL),y) OBJS += $(MAINOBJ) endif +ifeq ($(CONFIG_GRAPHICS_TRAVELER_ROMFSDEMO),y) +ROMFS_IMG = trv_romfs.img +ROMFS_HDR = trv_romfs.h +endif + ifeq ($(CONFIG_WINDOWS_NATIVE),y) BIN = ..\..\libapps$(LIBEXT) else @@ -90,10 +112,10 @@ ifeq ($(WINTOOL),y) INCDIROPT = -w endif -CFLAGS += ${shell $(INCDIR) $(INCDIROPT) "$(CC)" $(APPDIR)/graphics/traveler/include} +CFLAGS += ${shell $(INCDIR) $(INCDIROPT) "$(CC)" $(TRAVELER_INC)} -DEPPATH = --dep-path src -VPATH = src +DEPPATH = --dep-path . --dep-path src +VPATH = .:src # Common build @@ -106,7 +128,15 @@ $(AOBJS): %$(OBJEXT): %.S $(COBJS) $(MAINOBJ): %$(OBJEXT): %.c $(call COMPILE, $<, $@) -.built: $(OBJS) +ifeq ($(CONFIG_GRAPHICS_TRAVELER_ROMFSDEMO),y) +$(ROMFS_IMG): + $(Q) genromfs -f $@ -d $(TRAVELER_DEMO) -V "DemoWorld" + +$(ROMFS_HDR): $(ROMFS_IMG) + $(Q) (xxd -i $< | sed -e "s/^unsigned/static const unsigned/g" >$@) +endif + +.built: $(ROMFS_HDR) $(OBJS) $(call ARCHIVE, $(BIN), $(OBJS)) @touch .built @@ -140,6 +170,8 @@ depend: .depend clean: $(call DELFILE, .built) + $(call DELFILE, $(ROMFS_IMG)) + $(call DELFILE, $(ROMFS_HDR)) $(call CLEAN) distclean: clean diff --git a/graphics/traveler/include/trv_bitmaps.h b/graphics/traveler/include/trv_bitmaps.h index 82df0c145..948f55dce 100644 --- a/graphics/traveler/include/trv_bitmaps.h +++ b/graphics/traveler/include/trv_bitmaps.h @@ -1,5 +1,6 @@ /**************************************************************************** * apps/graphics/traveler/include/trv_bitmaps.h + * This file contains definitions for the texture bitmaps * * Copyright (C) 2014 Gregory Nutt. All rights reserved. * Author: Gregory Nutt @@ -46,12 +47,61 @@ * Pre-processor Definitions ****************************************************************************/ +#define BITMAP_WIDTH 64 +#define BITMAP_HEIGHT 64 +#define BITMAP_LOG2H 6 +#define BITMAP_SIZE (BITMAP_WIDTH * BITMAP_HEIGHT) +#define BITMAP_IMASK (BITMAP_HEIGHT-1) +#define BITMAP_JMASK (BITMAP_WIDTH-1) +#define BITMAP_JSHIFT 6 +#define BMICLIP(i) ((i) & BITMAP_IMASK) +#define BMJCLIP(i) ((i) & BITMAP_JMASK) +#define BMOFFSET(i,j) (((i) << BITMAP_JSHIFT) | (j)) +#define MAX_BITMAPS 256 + /**************************************************************************** * Public Types ****************************************************************************/ +struct trv_bitmap_s +{ + uint16_t w; + uint16_t h; + uint8_t log2h; + FAR trv_pixel_t *bm; +}; + +/**************************************************************************** + * Public Data + ****************************************************************************/ + +/* These point to the (allocated) bit map buffers for the even and odd + * bitmaps + */ + +extern FAR struct trv_bitmap_s *g_even_bitmaps[MAX_BITMAPS]; +#ifndef WEDIT +extern FAR struct trv_bitmap_s *g_odd_bitmaps[MAX_BITMAPS]; +#endif + +/* This is the maximum value + 1 of a texture code */ + +extern uint16_t g_trv_nbitmaps; + +/* These are the colors from the worldPalette which should used to rend + * the sky and ground + */ + +extern trv_pixel_t g_sky_color; +extern trv_pixel_t g_ground_color; + /**************************************************************************** * Public Function Prototypes ****************************************************************************/ +int trv_initialize_bitmaps(void); +void trv_release_bitmaps(void); +int trv_load_bitmapfile(FAR const char *bitmapfile); +FAR struct trv_bitmap_s *trv_read_texture(FAR char *filename); + #endif /* __APPS_GRAPHICS_TRAVELER_INCLUDE_TRV_BITMAPS_H */ diff --git a/graphics/traveler/include/trv_color.h b/graphics/traveler/include/trv_color.h index 9e8f8abb1..313001b7c 100644 --- a/graphics/traveler/include/trv_color.h +++ b/graphics/traveler/include/trv_color.h @@ -91,9 +91,9 @@ struct trv_color_rgb_s { - uint8_t red; - uint8_t green; - uint8_t blue; + uint8_t red; /* red component of color 0-63 */ + uint8_t green; /* green component of color 0-63 */ + uint8_t blue; /* blue component of color 0-63 */ }; struct trv_color_lum_s diff --git a/graphics/traveler/include/trv_doors.h b/graphics/traveler/include/trv_doors.h index 9179e2979..33499a656 100644 --- a/graphics/traveler/include/trv_doors.h +++ b/graphics/traveler/include/trv_doors.h @@ -42,6 +42,32 @@ #include "trv_types.h" +/**************************************************************************** + * Public Type Definitions + ****************************************************************************/ + +/* This structure describes the characteristics of the door which currently + * being opened. + */ + +struct trv_opendoor_s +{ + FAR struct trv_rect_data_s *rect; /* Points to the current door rectangle */ + uint8_t state; /* State of the door being opened */ + trv_coord_t zbottom; /* Z-Coordinate of the bottom of the door */ + trv_coord_t zdist; /* Distance which the door has moved */ + int16_t clock; /* This is clock which counts down the time + * remaining to keep the door open */ +}; + +/**************************************************************************** + * Public Data + ****************************************************************************/ + +/* This structure describes the door which is currently opening */ + +struct trv_opendoor_s g_opendoor; + /**************************************************************************** * Public Function Prototypes ****************************************************************************/ diff --git a/graphics/traveler/include/trv_fsutils.h b/graphics/traveler/include/trv_fsutils.h new file mode 100644 index 000000000..9a3fb15e5 --- /dev/null +++ b/graphics/traveler/include/trv_fsutils.h @@ -0,0 +1,53 @@ +/**************************************************************************** + * apps/graphics/traveler/include/trv_fsutils.h + * + * Copyright (C) 2014 Gregory Nutt. All rights reserved. + * Author: Gregory Nutt + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in + * the documentation and/or other materials provided with the + * distribution. + * 3. Neither the name NuttX nor the names of its contributors may be + * used to endorse or promote products derived from this software + * without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, + * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, + * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS + * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED + * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN + * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + * + ****************************************************************************/ + +#ifndef __APPS_GRAPHICS_TRAVELER_INCLUDE_TRV_FSUTILS_H +#define __APPS_GRAPHICS_TRAVELER_INCLUDE_TRV_FSUTILS_H + +/**************************************************************************** + * Included Files + ****************************************************************************/ + +#include "trv_types.h" + +#include + +/**************************************************************************** + * Public Function Prototypes + ****************************************************************************/ + +int16_t trv_read_decimal(FAR FILE *fp); + +#endif /* __APPS_GRAPHICS_TRAVELER_INCLUDE_TRV_FSUTILS_H */ diff --git a/graphics/traveler/include/trv_graphicfile.h b/graphics/traveler/include/trv_graphicfile.h new file mode 100644 index 000000000..dd4dd9db1 --- /dev/null +++ b/graphics/traveler/include/trv_graphicfile.h @@ -0,0 +1,78 @@ +/**************************************************************************** + * apps/graphics/traveler/include/trv_graphicfile.h + * + * Copyright (C) 2014 Gregory Nutt. All rights reserved. + * Author: Gregory Nutt + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in + * the documentation and/or other materials provided with the + * distribution. + * 3. Neither the name NuttX nor the names of its contributors may be + * used to endorse or promote products derived from this software + * without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, + * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, + * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS + * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED + * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN + * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + * + ****************************************************************************/ + +#ifndef __APPS_GRAPHICS_TRAVELER_INCLUDE_TRV_GRAPHICFILE_H +#define __APPS_GRAPHICS_TRAVELER_INCLUDE_TRV_GRAPHICFILE_H + +/**************************************************************************** + * Included Files + ****************************************************************************/ + +#include "trv_types.h" +#include "trv_color.h" + +/**************************************************************************** + * Pre-processor Definitions + ****************************************************************************/ + +/**************************************************************************** + * Public Type Definitions + ****************************************************************************/ + +struct trv_graphicfile_s +{ + uint16_t height; + uint16_t width; + int palette_entries; + long transparent_entry; + FAR struct trv_color_rgb_s *palette; + FAR uint8_t *bitmap; +}; + +/**************************************************************************** + * Public Data + ****************************************************************************/ + +/**************************************************************************** + * Public Function Prototypes + ****************************************************************************/ + +FAR struct trv_graphicfile_s *trv_graphicfile_new(void); +void trv_graphicfile_free(struct trv_graphicfile_s *gfile); +struct trv_color_rgb_s + trv_graphicfile_pixel(FAR struct trv_graphicfile_s *gfile, int x, int y); +struct trv_graphicfile_s *tvr_graphicfile_read(FAR char *filename); + +#endif /* __APPS_GRAPHICS_TRAVELER_INCLUDE_TRV_GRAPHICFILE_H */ diff --git a/graphics/traveler/include/trv_graphics.h b/graphics/traveler/include/trv_graphics.h index 69790e627..cb0684e9b 100644 --- a/graphics/traveler/include/trv_graphics.h +++ b/graphics/traveler/include/trv_graphics.h @@ -86,14 +86,15 @@ struct trv_graphics_info_s NXHANDLE hnx; /* The connection handle */ NXHANDLE bgwnd; /* Background window handle */ #else - trv_coord_t stride; /* Length of a line in bytes */ + trv_coord_t xoffset; /* Horizontal offset to start of data (pixels) */ + trv_coord_t yoffset; /* Vertical offset to start of data (rows) */ + trv_coord_t stride; /* Length of a line (bytes) */ #endif - trv_coord_t hwwidth; /* Display width (pixels) */ - trv_coord_t hwheight; /* Display height (rows) */ - trv_coord_t swwidth; /* Software render width (pixels) */ - trv_coord_t swheight; /* Software render height height (rows) */ - uint8_t vscale; /* Log2 vertical image scale factor */ - uint8_t hscale; /* Log2 horizontal image scale factor */ + trv_coord_t xres; /* Physical display width (pixels) */ + trv_coord_t yres; /* Physical display height (rows) */ + trv_coord_t imgwidth; /* Width of visible display region (bytes) */ + uint8_t xscale; /* Horizontal image scale factor */ + uint8_t yscale; /* Vertical image scale factor */ struct trv_palette_s palette; /* Color palette */ FAR dev_pixel_t *hwbuffer; /* Hardware frame buffer */ FAR trv_pixel_t *swbuffer; /* Software render buffer */ diff --git a/graphics/traveler/include/trv_paltable.h b/graphics/traveler/include/trv_paltable.h new file mode 100644 index 000000000..f7edec11c --- /dev/null +++ b/graphics/traveler/include/trv_paltable.h @@ -0,0 +1,85 @@ +/**************************************************************************** + * apps/graphics/traveler/include/trv_paltable.h + * + * Copyright (C) 2014 Gregory Nutt. All rights reserved. + * Author: Gregory Nutt + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in + * the documentation and/or other materials provided with the + * distribution. + * 3. Neither the name NuttX nor the names of its contributors may be + * used to endorse or promote products derived from this software + * without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, + * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, + * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS + * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED + * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN + * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + * + ****************************************************************************/ + +#ifndef __APPS_GRAPHICS_TRAVELER_INCLUDE_TRV_PALTABLE_H +#define __APPS_GRAPHICS_TRAVELER_INCLUDE_TRV_PALTABLE_H + +/**************************************************************************** + * Included Files + ****************************************************************************/ + +#include "trv_types.h" + +/**************************************************************************** + * Pre-processor + ****************************************************************************/ + +/* Here some palette-related definitions. */ + +#define PALETTE_SIZE 256 /* This is the number of colors in the palette */ +#define NUM_ZONES 16 /* This is the number of distance zones in the palette table */ + +/* Here are some macros used to access the palette table. */ + +#define PAL_SCALE_BITS 2 +#define GET_DISTANCE(x,y) ( (x) >= (y) ? ((x) + ((y) >>1)) : ((y) + ((x) >>1))) +#define GET_ZONE(x,y) (GET_DISTANCE(x,y) >> (sSHIFT+PAL_SCALE_BITS)) +#define GET_PALINDEX(d) ((d) >= NUM_ZONES ? (NUM_ZONES-1) : (d)) +#define GET_PALPTR(d) g_paltable[GET_PALINDEX(d)] + +/* This is a special version which is used by the texture logic. The + * texture engine used 8 bits of fraction in many of its calculation + */ + +#define GET_FZONE(x,y,n) (GET_DISTANCE(x,y) >> (sSHIFT+PAL_SCALE_BITS-(n))) + +/**************************************************************************** + * Public Data + ****************************************************************************/ + +/* This is the palette table which is used to adjust the texture values + * with distance + */ + +extern trv_pixel_t *g_paltable[NUM_ZONES]; + +/**************************************************************************** + * Public Function Prototypes + ****************************************************************************/ + +int trv_load_paltable(FAR const char *file); +void trv_release_paltable(void); + +#endif /* __APPS_GRAPHICS_TRAVELER_INCLUDE_TRV_PALTABLE_H */ diff --git a/graphics/traveler/include/trv_pcx.h b/graphics/traveler/include/trv_pcx.h new file mode 100644 index 000000000..81a532cab --- /dev/null +++ b/graphics/traveler/include/trv_pcx.h @@ -0,0 +1,87 @@ +/**************************************************************************** + * apps/graphics/traveler/include/trv_pcx.h + * This is the header file associated with trv_pcx.c + * + * Copyright (C) 2014 Gregory Nutt. All rights reserved. + * Author: Gregory Nutt + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in + * the documentation and/or other materials provided with the + * distribution. + * 3. Neither the name NuttX nor the names of its contributors may be + * used to endorse or promote products derived from this software + * without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, + * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, + * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS + * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED + * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN + * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + * + ****************************************************************************/ + +#ifndef __APPS_GRAPHICS_TRAVELER_INCLUDE_TRV_PCX_H +#define __APPS_GRAPHICS_TRAVELER_INCLUDE_TRV_PCX_H + +/**************************************************************************** + * Included Files + ****************************************************************************/ + +#include "trv_types.h" + +#include + +/**************************************************************************** + * Pre-processor Definitions + ****************************************************************************/ + +#define SIZEOF_PCX_HEADER 128 + +/**************************************************************************** + * Public Type Definitions + ****************************************************************************/ + +struct pcx_header_s +{ + char manufacturer; + char version; + char encoding; + char bits_per_pixel; + int16_t x, y; + int16_t width, height; + int16_t horz_res; + int16_t vert_res; + char ega_palette[48]; + char reserved; + char num_color_planes; + int16_t bytes_per_line; + int16_t palette_type; + char padding[58]; +}; + +/**************************************************************************** + * Public Data + ****************************************************************************/ + +/**************************************************************************** + * Public Function Prototypes + ****************************************************************************/ + +FAR struct trv_graphicfile_s *trv_load_pcx(FAR FILE *fp, + FAR const char *filename); + +#endif /* __APPS_GRAPHICS_TRAVELER_INCLUDE_TRV_PCX_H */ diff --git a/graphics/traveler/include/trv_plane.h b/graphics/traveler/include/trv_plane.h new file mode 100644 index 000000000..936fc4e9e --- /dev/null +++ b/graphics/traveler/include/trv_plane.h @@ -0,0 +1,163 @@ +/**************************************************************************** + * apps/graphics/traveler/include/trv_plane.h + * This file contains definitions for the world planes + * + * Copyright (C) 2014 Gregory Nutt. All rights reserved. + * Author: Gregory Nutt + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in + * the documentation and/or other materials provided with the + * distribution. + * 3. Neither the name NuttX nor the names of its contributors may be + * used to endorse or promote products derived from this software + * without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, + * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, + * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS + * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED + * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN + * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + * + ****************************************************************************/ + +#ifndef __APPS_GRAPHICS_TRAVELER_INCLUDE_TRV_PLANE_H +#define __APPS_GRAPHICS_TRAVELER_INCLUDE_TRV_PLANE_H + +/**************************************************************************** + * Included Files + ****************************************************************************/ + +#include "trv_types.h" +#include + +/**************************************************************************** + * Pre-processor Definitions + ****************************************************************************/ + +/* These are bit-field definitions for the struct trv_rect_data_s attribute field + * STATIC settings + */ + +#define SHADED_PLANE 0x01 +#define TRANSPARENT_PLANE 0x02 +#define DOOR_PLANE 0x04 + +/* DYNAMIC settings */ + +#define OPEN_DOOR_PLANE 0x08 +#define MOVING_DOOR_PLANE 0x10 + +/* Legal values of texture scaling in struct trv_rect_data_s */ + +#define ONEX_SCALING 0 +#define TW0X_SCALING 1 +#define FOURX_SCALING 2 +#define MAXX_SCALING 2 + +/* These macros are used to test the various texture attributes */ + +#define IS_NORMAL(r) (((r)->attribute & ~SHADED_PLANE) == 0) +#define IS_SHADED(r) (((r)->attribute & SHADED_PLANE) != 0) +#define IS_TRANSPARENT(r) (((r)->attribute & TRANSPARENT_PLANE) != 0) +#define IS_DOOR(r) (((r)->attribute & DOOR_PLANE) != 0) +#define IS_OPEN_DOOR(r) (((r)->attribute & OPEN_DOOR_PLANE) != 0) +#define IS_CLOSED_DOOR(r) ((IS_DOOR(R)) && (!IS_OPEN_DOOR(r))) +#define IS_MOVING_DOOR(r) (((r)->attribute & MOVING_DOOR_PLANE) != 0) +#define IS_PASSABLE(r) IS_OPEN_DOOR(r) + +/**************************************************************************** + * Public Types + ****************************************************************************/ + +typedef uint8_t attrib_t; /* Max attributes = 8 */ + +struct trv_rect_data_s +{ + trv_coord_t plane; /* defines the plane that the rect lies in */ + trv_coord_t hstart; /* defines the starting "horizontal" position */ + trv_coord_t hend; /* defines the ending "horizontal" position */ + trv_coord_t vstart; /* defines the starting "vertical" position */ + trv_coord_t vend; /* defines the ending "vertical" position */ + attrib_t attribute; /* bit-encoded attributes of the plane */ + uint8_t texture; /* defines the texture that should be applied */ + uint8_t scale; /* defines the scaling of the texture */ +}; +#define RESIZEOF_TRVRECTDATA_T 13 + +struct trv_rect_list_s +{ + struct trv_rect_list_s *flink; /* points at next rectangle in a list */ + struct trv_rect_list_s *blink; /* points at previous rectangle in a list */ + struct trv_rect_data_s d; /* the data which defines the rectangle */ +}; + +struct trv_rect_head_s +{ + struct trv_rect_list_s *head; /* points to the start of the list */ + struct trv_rect_list_s *tail; /* points to the end of the list */ +}; + +struct trv_planefile_header_s +{ + uint16_t nxrects; + uint16_t nyrects; + uint16_t nzrects; +}; +#define SIZEOF_TRVPLANEFILEHEADER_T 6 + +/**************************************************************************** + * Public Data + ****************************************************************************/ + +/* The is the world!!! The world is described by lists of rectangles, one + * for each of the X, Y, and Z planes. + */ + +extern struct trv_rect_head_s g_xplane; /* list of X=plane rectangles */ +extern struct trv_rect_head_s g_yplane; /* list of Y=plane rectangles */ +extern struct trv_rect_head_s g_zplane; /* list of Z=plane rectangles */ + +/* "Deallocated" planes are retained in a free list */ + +extern struct trv_rect_list_s *g_rect_freelist; + +/**************************************************************************** + * Public Function Prototypes + ****************************************************************************/ + +/* Plane list management */ + +int trv_initialize_planes(void); +void trv_add_plane(FAR struct trv_rect_list_s *rect, + FAR struct trv_rect_head_s *list); +void trv_move_plane(FAR struct trv_rect_list_s *rect, + FAR struct trv_rect_head_s *destlist, + FAR struct trv_rect_head_s *srclist); +void trv_merge_planelists(FAR struct trv_rect_head_s *outlist, + FAR struct trv_rect_head_s *inlist); +void trv_release_planes(void); + +/* Plane memory management */ + +FAR struct trv_rect_list_s *trv_new_plane(void); + +/* Plane file management */ + +int trv_load_planefile(FAR const char *wldfile); +int trv_save_planes(const char *wldfile); + +#endif /* __APPS_GRAPHICS_TRAVELER_INCLUDE_TRV_PLANE_H */ diff --git a/graphics/traveler/include/trv_pov.h b/graphics/traveler/include/trv_pov.h index 2edc5b7a2..70038049b 100644 --- a/graphics/traveler/include/trv_pov.h +++ b/graphics/traveler/include/trv_pov.h @@ -49,7 +49,7 @@ /* This structure defines the current camera position of the player's eyes */ -extern struct trv_camera_s g_trv_player; +extern struct trv_camera_s g_player; /**************************************************************************** * Public Function Prototypes diff --git a/graphics/traveler/include/trv_rayavoid.h b/graphics/traveler/include/trv_rayavoid.h index 8778ed7de..08214cbc8 100644 --- a/graphics/traveler/include/trv_rayavoid.h +++ b/graphics/traveler/include/trv_rayavoid.h @@ -55,13 +55,19 @@ * Public Function Prototypes ****************************************************************************/ -extern trv_coord_t trv_rayclip_player_xmotion(FAR struct trv_camera_s *pov, - trv_coord_t dist, int16_t yaw, - trv_coord_t height); -extern trv_coord_t trv_rayclip_player_ymotion(FAR struct trv_camera_s *pov, - trv_coord_t dist, int16_t yaw, - trv_coord_t height); -extern trv_coord_t trv_ray_adjust_zpos(FAR struct trv_camera_s *pov, +trv_coord_t trv_rayclip_player_xmotion(FAR struct trv_camera_s *pov, + trv_coord_t dist, int16_t yaw, trv_coord_t height); +trv_coord_t trv_rayclip_player_ymotion(FAR struct trv_camera_s *pov, + trv_coord_t dist, int16_t yaw, + trv_coord_t height); +trv_coord_t trv_ray_adjust_zpos(FAR struct trv_camera_s *pov, + trv_coord_t height); +FAR struct trv_rect_data_s *trv_ray_test_xplane(FAR struct trv_camera_s *pov, + trv_coord_t dist, int16_t yaw, + trv_coord_t height); +FAR struct trv_rect_data_s *trv_ray_test_yplane(FAR struct trv_camera_s *pov, + trv_coord_t dist, int16_t yaw, + trv_coord_t height); #endif /* __APPS_GRAPHICS_TRAVELER_INCLUDE_TRV_RAYAVOID_H */ diff --git a/graphics/traveler/include/trv_raycast.h b/graphics/traveler/include/trv_raycast.h new file mode 100644 index 000000000..c48d1fca0 --- /dev/null +++ b/graphics/traveler/include/trv_raycast.h @@ -0,0 +1,169 @@ +/**************************************************************************** + * apps/graphics/traveler/include/trv_raycast.h + * This is the header file associated with trv_raycast.c + * + * Copyright (C) 2014 Gregory Nutt. All rights reserved. + * Author: Gregory Nutt + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in + * the documentation and/or other materials provided with the + * distribution. + * 3. Neither the name NuttX nor the names of its contributors may be + * used to endorse or promote products derived from this software + * without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, + * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, + * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS + * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED + * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN + * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + * + ****************************************************************************/ + +#ifndef __APPS_GRAPHICS_TRAVELER_INCLUDE_TRV_RAYCAST_H +#define __APPS_GRAPHICS_TRAVELER_INCLUDE_TRV_RAYCAST_H + +/**************************************************************************** + * Included Files + ****************************************************************************/ + +#include "trv_types.h" + +/**************************************************************************** + * Pre-processor Definitions + ****************************************************************************/ + +/* The following definitions define the world view window in terms of the + * 320x200 display coordinates. + */ + +#define WINDOW_LEFT 0 /* Offset (in bytes) to the left side + * of the world window */ +#define WINDOW_WIDTH 320 /* Width of the world window */ +#define WINDOW_TOP 0 /* Offset (in multiples of TRV_SCREEN_WIDTH) + * to the top of world window */ +#define WINDOW_HEIGHT 200 /* height of the world window */ +#define WINDOW_MIDDLE 100 /* Center or horizon of the world window */ + +/* This calculation tends to overflow unless it is cast carefully */ + +#define WINDOW_SIZE (((int32_t)WINDOW_WIDTH)*((int32_t)WINDOW_HEIGHT)) + +/* This is the biggest distance that can be represented with int16_t */ + +#define TRV_INFINITY 0x7fff + +/* The actual size of the image will be determined by the + * both the WINDOW size as well as the GULP size + */ + +#define NUMBER_HGULPS (WINDOW_WIDTH/HGULP_SIZE) +#define IMAGE_WIDTH (HGULP_SIZE*NUMBER_HGULPS) +#define IMAGE_LEFT (((WINDOW_WIDTH-IMAGE_WIDTH) >> 1) + WINDOW_LEFT) +#define IMAGE_HEIGHT WINDOW_HEIGHT +#define IMAGE_TOP WINDOW_TOP + +/* This defines the number of pixels (in each direction) with will be + * processed on each pass through the innermost ray casting loop + */ + +#define HGULP_SIZE 7 +#define VGULP_SIZE 16 + +/* The following define the various meanings of hit.type */ + +#define NO_HIT 0x00 +#define FRONT_HIT 0x00 +#define BACK_HIT 0x01 +#define FB_MASK 0x01 +#define X_HIT 0x00 +#define Y_HIT 0x02 +#define Z_HIT 0x04 +#define XYZ_MASK 0x06 + +#define IS_FRONT_HIT(h) (((h)->type & FB_MASK) == FRONT_HIT) +#define IS_BACK_HIT(h) (((h)->type & FB_MASK) == BACK_HIT) +#define IS_XRAY_HIT(h) (((h)->type & XYZ_MASK) == X_HIT) +#define IS_YRAY_HIT(h) (((h)->type & XYZ_MASK) == Y_HIT) +#define IS_ZRAY_HIT(h) (((h)->type & XYZ_MASK) == Z_HIT) + +#define MK_HIT_TYPE(fb,xyz) ((fb)|(xyz)) + +/**************************************************************************** + * Public Types + ****************************************************************************/ + +/* This structure provides the return values for each ray caster. For + * performance reasons, the size of this structure should be an even + * power of two (makes index calculation faster). + */ + +struct trv_rect_data_s; +struct trv_raycast_s +{ + /* 'rect' points to the rectangle that was hit (if any) */ + + FAR struct trv_rect_data_s *rect; + + uint8_t type; /* Type of hit: X/Y/Z cast, front/back */ + uint8_t unused2; /* Padding to force power of two size */ + int16_t xpos; /* Horizontal position on surface of the hit */ + int16_t ypos; /* Vertical position on surface of the hit */ + int16_t xdist; /* X distance to the hit */ + int16_t ydist; /* Y distance to the hit */ + int16_t zdist; /* Z distance to the hit (not used) */ +}; + +/**************************************************************************** + * Public Data + ****************************************************************************/ + +/* This structure holds pre-calculated pitch data for all ray casts. NOTE: + * would be a performance improvement if this structure is an even power of + *two in size (don't have to multiply to calculate indices) + */ + +/* The following array describes the hits from X/Y/Z-ray casting for the + * current HGULP_SIZE x VGULP_SIZE cell + */ + +extern struct trv_raycast_s g_ray_hit[VGULP_SIZE][HGULP_SIZE+1]; + +/* This structure points to the double buffer row corresponding to the + * pitch angle + */ + +extern uint8_t *g_buffer_row[VGULP_SIZE]; + +/* The is the "column" offset in g_buffer_row for the current cell being + * operated on. This value is updated in a loop by trv_raycaster. + */ + +extern int16_t g_cell_column; + +/* This structure holds the parameters used in the current ray cast */ + +extern struct trv_camera_s g_camera; + +/**************************************************************************** + * Public Function Prototypes + ****************************************************************************/ + +void trv_raycast(int16_t pitch, int16_t yaw, int16_t screenyaw, + FAR struct trv_raycast_s *result); + +#endif /* __APPS_GRAPHICS_TRAVELER_INCLUDE_TRV_RAYCAST_H */ diff --git a/graphics/traveler/include/trv_raycntl.h b/graphics/traveler/include/trv_raycntl.h index ffcf2b6d3..7726c2006 100644 --- a/graphics/traveler/include/trv_raycntl.h +++ b/graphics/traveler/include/trv_raycntl.h @@ -41,12 +41,16 @@ ****************************************************************************/ #include "trv_types.h" -#include "trv_graphics.h" /**************************************************************************** * Pre-processor Definitions ****************************************************************************/ +/* This is the size of the buffer supported by the ray caster */ + +#define TRV_SCREEN_WIDTH 320 +#define TRV_SCREEN_HEIGHT 200 + /**************************************************************************** * Public Types ****************************************************************************/ @@ -55,7 +59,11 @@ * Public Function Prototypes ****************************************************************************/ +struct trv_camera_s; +struct trv_graphics_info_s; + void trv_raycaster(FAR struct trv_camera_s *player, FAR struct trv_graphics_info_s *ginfo); +uint8_t trv_get_texture(uint8_t row, uint8_t col); #endif /* __APPS_GRAPHICS_TRAVELER_INCLUDE_TRV_RAYCNTL_H */ diff --git a/graphics/traveler/include/trv_rayprune.h b/graphics/traveler/include/trv_rayprune.h new file mode 100644 index 000000000..ec35805e1 --- /dev/null +++ b/graphics/traveler/include/trv_rayprune.h @@ -0,0 +1,62 @@ +/**************************************************************************** + * apps/graphics/traveler/include/trv_rayprune.h + * + * Copyright (C) 2014 Gregory Nutt. All rights reserved. + * Author: Gregory Nutt + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in + * the documentation and/or other materials provided with the + * distribution. + * 3. Neither the name NuttX nor the names of its contributors may be + * used to endorse or promote products derived from this software + * without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, + * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, + * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS + * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED + * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN + * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + * + ****************************************************************************/ + +#ifndef __APPS_GRAPHICS_TRAVELER_INCLUDE_TRV_RAYPRUNE_H +#define __APPS_GRAPHICS_TRAVELER_INCLUDE_TRV_RAYPRUNE_H + +/**************************************************************************** + * Included Files + ****************************************************************************/ + +#include "trv_types.h" + +/**************************************************************************** + * Public Data + ****************************************************************************/ + +extern struct trv_rect_head_s g_ray_xplane; /* List of X=plane rectangles */ +extern struct trv_rect_head_s g_ray_yplane; /* List of Y=plane rectangles */ +extern struct trv_rect_head_s g_ray_zplane; /* List of Z=plane rectangles */ + +/**************************************************************************** + * Public Function Prototypes + ****************************************************************************/ + +void trv_ray_yawprune(int16_t yawstart, int16_t yawend); +void trv_ray_pitchprune(int16_t pitchstart, int16_t pitchend); +void trv_ray_pitchunprune(void); +void trv_ray_yawunprune(void); + +#endif /* __APPS_GRAPHICS_TRAVELER_INCLUDE_TRV_RAYPRUNE_H */ diff --git a/graphics/traveler/include/trv_rayrend.h b/graphics/traveler/include/trv_rayrend.h index 189e17e46..5ae77f59a 100644 --- a/graphics/traveler/include/trv_rayrend.h +++ b/graphics/traveler/include/trv_rayrend.h @@ -41,12 +41,37 @@ ****************************************************************************/ #include "trv_types.h" -#include "trv_world.h" /**************************************************************************** * Pre-processor Definitions ****************************************************************************/ +/* The following macro calculates the texture index for the given the + * horizontal position (h), the vertical position (v) and the log2 bitmap + * size (s). + */ + +#define TMASK(s) ((1 << (s)) - 1) +#define TFIX(x,s) ((x) << (s)) +#define TFRAC(x,m) ((x) & (m)) +#define TNDX(h,v,s,m) (TFIX( TFRAC( (~(v)), m ), s) + TFRAC( h, m ) ) + +/* Here is a macro to extract one pixel from a WALL texture. This is used + * by the raycasters to determine a hit in a transparent wall hit an + * opaque portion of the texture + */ + +#define GET_FRONT_PIXEL(r,h,v) \ + trv_get_rectpixel(h,v,g_even_bitmaps[ (r)->texture ], (r)->scale) +#define GET_BACK_PIXEL(r,h,v) \ + trv_get_rectpixel(h,v,g_odd_bitmaps[ (r)->texture ], (r)->scale) + +/* This is special value of a pixel in a "grandparent" rectangle which means + * that the pixel is transparent. + */ + +#define INVISIBLE_PIXEL 0 + /**************************************************************************** * Public Types ****************************************************************************/ @@ -55,6 +80,17 @@ * Public Function Prototypes ****************************************************************************/ -void trv_rend_backdrop (struct trv_camera_s *camera); +struct trv_camera_s; +struct trv_graphics_info_s; +struct trv_bitmap_s; + +void trv_rend_backdrop(FAR struct trv_camera_s *camera, + FAR struct trv_graphics_info_s *ginfo); +void trv_rend_cell(uint8_t row, uint8_t col, uint8_t height, uint8_t width); +void trv_rend_row(uint8_t row, uint8_t col, uint8_t width); +void trv_rend_column(uint8_t row, uint8_t col, uint8_t height); +void trv_rend_pixel(uint8_t row, uint8_t col); +trv_pixel_t trv_get_rectpixel(int16_t hPos, int16_t vPos, + FAR struct trv_bitmap_s *bmp, uint8_t scale); #endif /* __APPS_GRAPHICS_TRAVELER_INCLUDE_TRV_RAYEND_H */ diff --git a/graphics/traveler/include/trv_trigtbl.h b/graphics/traveler/include/trv_trigtbl.h index deac5a0fb..c8479d87f 100644 --- a/graphics/traveler/include/trv_trigtbl.h +++ b/graphics/traveler/include/trv_trigtbl.h @@ -1,5 +1,7 @@ /**************************************************************************** * apps/graphics/traveler/include/trv_trigtbl.h + * This file defines the fixed precision math environment and look-up tables + * for trigonometric functions. * * Copyright (C) 2014 Gregory Nutt. All rights reserved. * Author: Gregory Nutt @@ -45,6 +47,7 @@ /**************************************************************************** * Pre-processor Definitions ****************************************************************************/ +/* Angles *******************************************************************/ /* These are definitions of commonly used angles. */ @@ -69,6 +72,126 @@ #define ANGLE_270 1440 #define ANGLE_360 1920 +/* This is the angular change made with each column of the ray caster */ +/* This is (2048/360 units/degree) * 59.94 (degrees) / (320 columns) */ + +#define VIDEO_COLUMN_ANGLE 1 +#define VIDEO_ROW_ANGLE 1 + +/* Fixed precision definitions **********************************************/ + +/* SMALL precision (6 bits past binary point) */ +/* This occurs frequently because each CELL is 64x64 */ + +#define sUNITY 64 +#define sHALF 32 +#define sQUARTER 16 +#define sSHIFT 6 +#define sMASK 63 + +#define sTRUNC(a) ((a) >> sSHIFT) +#define sROUND(a) (((a) + sHALF) >> sSHIFT) +#define sFIX(a) ((a) << sSHIFT) +#define sSNAP(a) ((a) & (~sMASK)) +#define sFRAC(a) ((a) & sMASK) + +#define sMUL(a,b) (((int32_t)(a) * (int32_t)(b)) >> sSHIFT ) +#define sDIV(a,b) (((int32_t)(a) << sSHIFT) / (b)) +#define sFLOAT(a) ((float)(a) / (float)sUNITY) + +/* DOUBLE precision (12 bits past binary point) */ +/* This precision results when two SMALL precision numbers are multiplied */ + +#define dUNITY 4096 +#define dHALF 2048 +#define dSHIFT 12 +#define dMASK 4095 + +#define dTRUNC(a) ((a) >> dSHIFT) +#define dROUND(a) (((a) + dHALF) >> dSHIFT) +#define dFIX(a) ((a) << dSHIFT) +#define dSNAP(a) ((a) & (~dMASK)) +#define dFRAC(a) ((a) & dMASK) + +/* TRIPLE precision (18 bits past binary point) */ +/* This precision results when a SMALL and a DOUBLE precision number + * are multiplied + */ + +#define tSHIFT 18 + +/* QUAD precision (24 bits past binary point) */ +/* This precision results when two DOUBLE precision numbers are multiplied */ + +#define qSHIFT 24 + +/* BIG precision (16 bits past binary point) */ +/* This is convenient precision because it is easy to extract the integer + * portion without shifting or masking + */ + +#define bUNITY 65536 +#define bHALF 32768 +#define bSHIFT 16 +#define bMASK 65535 + +/* Conversions between SMALL, DOUBLE, TRIPLE and BIG precision */ + +#define sTOd(a) ((a) << (dSHIFT-sSHIFT)) +#define sTOb(a) ((a) << (bSHIFT-sSHIFT)) +#define dTOs(a) ((a) >> (dSHIFT-sSHIFT)) +#define dTOb(a) ((a) << (bSHIFT-dSHIFT)) +#define tTOs(a) ((a) >> (tSHIFT-sSHIFT)) +#define tTOd(a) ((a) >> (tSHIFT-dSHIFT)) +#define tTOb(a) ((a) >> (tSHIFT-bSHIFT)) +#define qTOd(a) ((a) >> (qSHIFT-dSHIFT)) +#define qTOb(a) ((a) >> (qSHIFT-bSHIFT)) +#define bTOs(a) ((a) >> (bSHIFT-sSHIFT)) + +/* These are general math macros that have nothing to do with fixed precision */ + +#define ABS(a) ((a) < 0 ? -(a) : (a)) +#define MIN(a,b) ((a) < (b) ? (a) : (b)) +#define MAX(a,b) ((a) > (b) ? (a) : (b)) + +/* Trigonometry *************************************************************/ +/* Because COS(x) = SIN(x + HALFPI) and COT(x) = TAN(90-x), the following + * provide fast conversions to get cosines from the g_sin_table's and + * cotangents form the g_tan_table. + */ + +#define g_cot_table(x) g_tan_table[PI+HALFPI-(x)] +#define g_cos_table ((int16_t*)&g_sin_table[HALFPI]) +#define g_sec_table ((int32_t*)&g_csc_table[HALFPI]) + +/* Here are some MACROs to make life easier */ +/* The following extend the range of the table to all positive angles */ + +#define TAN(x) ((x)>=(PI+HALFPI) ? g_tan_table[(x)-PI] : g_tan_table[x]) + +/**************************************************************************** + * Public Types + ****************************************************************************/ +/* This structure is useful for manipulating BIG precision types from + * C code (NOTE: The following union assumes LITTLE ENDIAN!). + */ + +struct trv_bigfp_s +{ + uint16_t f; + int16_t i; +}; + +/**************************************************************************** + * Public Data + ****************************************************************************/ + +/* Here are declarations for the trig tables */ + +extern const int32_t g_tan_table[PI+HALFPI+1]; +extern const int16_t g_sin_table[TWOPI+HALFPI+1]; +extern const int32_t g_csc_table[TWOPI+HALFPI+1]; + /**************************************************************************** * Public Function Prototypes ****************************************************************************/ diff --git a/graphics/traveler/include/trv_types.h b/graphics/traveler/include/trv_types.h index 2eb9f2618..3a5fb4247 100644 --- a/graphics/traveler/include/trv_types.h +++ b/graphics/traveler/include/trv_types.h @@ -65,6 +65,7 @@ typedef uint8_t trv_pixel_t; /* Width of one pixel in rendering phase */ typedef int16_t trv_coord_t; /* Contains one display coordinate */ + /* Max world size is +/- 65536/64 = 1024 */ /**************************************************************************** * Public Function Prototypes diff --git a/graphics/traveler/include/trv_world.h b/graphics/traveler/include/trv_world.h index 3461fd792..97b6c7443 100644 --- a/graphics/traveler/include/trv_world.h +++ b/graphics/traveler/include/trv_world.h @@ -1,5 +1,6 @@ /**************************************************************************** * apps/graphics/traveler/include/trv_world.h + * This file contains definitions for the world model * * Copyright (C) 2014 Gregory Nutt. All rights reserved. * Author: Gregory Nutt diff --git a/graphics/traveler/src/trv_bitmapfile.c b/graphics/traveler/src/trv_bitmapfile.c new file mode 100644 index 000000000..ea39eaae9 --- /dev/null +++ b/graphics/traveler/src/trv_bitmapfile.c @@ -0,0 +1,232 @@ +/******************************************************************************* + * apps/graphics/traveler/src/trv_bitmapfile.c + * This file contains the logic which loads texture bitmaps + * + * Copyright (C) 2014 Gregory Nutt. All rights reserved. + * Author: Gregory Nutt + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in + * the documentation and/or other materials provided with the + * distribution. + * 3. Neither the name NuttX nor the names of its contributors may be + * used to endorse or promote products derived from this software + * without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, + * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, + * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS + * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED + * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN + * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + * + ****************************************************************************/ + +/**************************************************************************** + * Included files + ****************************************************************************/ + +#include "trv_types.h" +#include "trv_mem.h" +#include "trv_world.h" +#include "trv_bitmaps.h" +#include "trv_fsutils.h" + +#include +#include +#include + +/**************************************************************************** + * Private Functions + ****************************************************************************/ + +/**************************************************************************** + * Name: trv_read_filename + * + * Description: + * Read a file name from the input stream + * + ***************************************************************************/ + +static int trv_read_filename(FAR FILE *fp, FAR char *filename) +{ + int16_t nbytes; + int ch; + + /* Skip over any leading whitespace */ + + do + { + ch = getc(fp); + } + while (isspace(ch)); + + if (ch == EOF) + { + return -ENOENT; + } + + /* Get the file name from the file */ + + nbytes = 0; + for (;;) + { + /* Everything up to the next whitespace must be the filename */ + + if (isspace(ch) || ch == EOF) + { + /* End of the string. Don't forget the ASCIIZ terminator */ + + filename[nbytes] = '\0'; + break; + } + + /* Make sure that the file name is not too large */ + + else if (nbytes >= FILENAME_MAX) + { + return -E2BIG; + } + + /* Add the new character to the file name */ + + else + { + filename[nbytes] = (char)ch; + nbytes++; + } + + /* Get the character for the next time through the loop. Every file + * name should be terminated with a space or a new line (or, possibily, + * EOF) + */ + + ch = getc(fp); + } + + return OK; +} + +/**************************************************************************** + * Name: trv_load_bitmaps + * + * Description: + * This function loads the world data from the input file + * + ***************************************************************************/ + +static int trv_load_bitmaps(FAR FILE *fp) +{ + char filename[FILENAME_MAX]; + int mapndx; + int ret = OK; + + /* Discard any bitmaps that we may currently have buffered */ + + trv_release_bitmaps(); + + /* Get the number of bitmaps in the bitmap file */ + + g_trv_nbitmaps = trv_read_decimal(fp); + if (g_trv_nbitmaps >= MAX_BITMAPS) + { + return -E2BIG; + } + + /* Read the colors used to rend the sky and ground */ + + g_sky_color = trv_read_decimal(fp); + g_ground_color = trv_read_decimal(fp); + + /* Load each bitmap file */ + + for (mapndx = 0; mapndx < g_trv_nbitmaps && ret == OK; mapndx++) + { + /* Load the even bitmap */ + /* Get the name of the file which contains the even bitmap */ + + ret = trv_read_filename(fp, filename); + if (ret < 0) + { + return ret; + } + + g_even_bitmaps[mapndx] = trv_read_texture(filename); + if (!g_even_bitmaps[mapndx]) + { + return -EIO; + } + + /* Load the odd bitmap */ + /* Get the name of the file which contains the odd bitmap */ + + ret = trv_read_filename(fp, filename); + if (ret < 0) + { + return ret; + } + +#ifndef WEDIT + g_odd_bitmaps[mapndx] = trv_read_texture(filename); + if (!g_odd_bitmaps[mapndx]) + { + return -EIO; + } +#endif + } + + return ret; +} + +/**************************************************************************** + * Public Functions + ***************************************************************************/ + +/**************************************************************************** + * Name: trv_load_bitmapfile + * + * Description: + * This function opens the input file and loads the world data from it + * + ***************************************************************************/ + +int trv_load_bitmapfile(FAR const char *bitmapfile) +{ + FAR FILE *fp; + int ret; + + /* Open the file which contains the names of all of the bitmap files */ + + fp = fopen(bitmapfile, "r"); + if (!fp) + { + int errcode = errno; + return -errcode; + } + + /* Load all of the bitmaps */ + + ret = trv_load_bitmaps(fp); + if (ret < 0) + { + trv_release_bitmaps(); + } + + /* We are all done with the file, so close it */ + + fclose(fp); + return ret; +} + diff --git a/graphics/traveler/src/trv_bitmaps.c b/graphics/traveler/src/trv_bitmaps.c new file mode 100644 index 000000000..c9518ab60 --- /dev/null +++ b/graphics/traveler/src/trv_bitmaps.c @@ -0,0 +1,152 @@ +/******************************************************************************* + * apps/graphics/traveler/src/trv_bitmaps.c + * This file contains low-level texture bitmap logic + * + * Copyright (C) 2014 Gregory Nutt. All rights reserved. + * Author: Gregory Nutt + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in + * the documentation and/or other materials provided with the + * distribution. + * 3. Neither the name NuttX nor the names of its contributors may be + * used to endorse or promote products derived from this software + * without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, + * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, + * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS + * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED + * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN + * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + * + ****************************************************************************/ + +/**************************************************************************** + * Included files + ****************************************************************************/ + +#include "trv_types.h" +#include "trv_mem.h" +#include "trv_bitmaps.h" + +/**************************************************************************** + * Pre-processor Definitions + ****************************************************************************/ + +/**************************************************************************** + * Public Data + ****************************************************************************/ + +/* These point to the (allocated) bit map buffers for the even and odd + * bitmaps + */ + +FAR struct trv_bitmap_s *g_even_bitmaps[MAX_BITMAPS]; +#ifndef WEDIT +FAR struct trv_bitmap_s *g_odd_bitmaps[MAX_BITMAPS]; +#endif + +/* This is the maximum value + 1 of a texture code. */ + +uint16_t g_trv_nbitmaps; + +/* These are the colors from the world palette which should used to rend + * the sky and ground + */ + +trv_pixel_t g_sky_color; +trv_pixel_t g_ground_color; + +/**************************************************************************** + * Private Functions + ****************************************************************************/ + +/************************************************************************* + * Name: trv_free_texture + * + * Description: + * Free both the bitmap and the bitmap container + * + ************************************************************************/ + +static void trv_free_texture(FAR struct trv_bitmap_s *bitmap) +{ + if (bitmap->bm) + { + trv_free(bitmap->bm); + } + + trv_free(bitmap); +} + +/**************************************************************************** + * Public Functions + ****************************************************************************/ + +/**************************************************************************** + * Name: trv_initialize_bitmaps + * + * Description: + * + ***************************************************************************/ + +int trv_initialize_bitmaps(void) +{ + int i; + + for (i = 0; i < MAX_BITMAPS; i++) + { + g_even_bitmaps[i] = NULL; +#ifndef WEDIT + g_odd_bitmaps[i] = NULL; +#endif + } + + g_trv_nbitmaps = 0; + return OK; +} + +/************************************************************************* + * Name: trv_release_bitmaps + * + * Description: + * This function deallocates all bitmaps. + * + ************************************************************************/ + +void trv_release_bitmaps(void) +{ + int i; + + for (i = 0; i < MAX_BITMAPS; i++) + { + if (g_even_bitmaps[i]) + { + trv_free_texture(g_even_bitmaps[i]); + g_even_bitmaps[i] = NULL; + } + +#ifndef WEDIT + if (g_odd_bitmaps[i]) + { + trv_free_texture(g_odd_bitmaps[i]); + g_odd_bitmaps[i] = NULL; + } +#endif + } + + g_trv_nbitmaps = 0; +} diff --git a/graphics/traveler/src/trv_doors.c b/graphics/traveler/src/trv_doors.c new file mode 100644 index 000000000..3fe3785e0 --- /dev/null +++ b/graphics/traveler/src/trv_doors.c @@ -0,0 +1,322 @@ +/******************************************************************************* + * apps/graphics/traveler/src/trv_doors.c + * This file contains the logic which manages world door logic. + * + * Copyright (C) 2014 Gregory Nutt. All rights reserved. + * Author: Gregory Nutt + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in + * the documentation and/or other materials provided with the + * distribution. + * 3. Neither the name NuttX nor the names of its contributors may be + * used to endorse or promote products derived from this software + * without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, + * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, + * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS + * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED + * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN + * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + * + ****************************************************************************/ + +/**************************************************************************** + * Included Files + ****************************************************************************/ + +#include "trv_types.h" +#include "trv_input.h" +#include "trv_pov.h" +#include "trv_plane.h" +#include "trv_world.h" +#include "trv_rayavoid.h" +#include "trv_doors.h" + +/**************************************************************************** + * Pre-processor Definitions + ****************************************************************************/ + +/* This is the door opening rate */ + +#define DOOR_ZSTEP 15 +#define DOOR_OPEN_WAIT 30 + +/**************************************************************************** + * Private Type Declarations + ***************************************************************************/ + +/* These are possible values for the g_opendoor state variable */ + +enum trv_door_state_e +{ + DOOR_IDLE = 0, /* No door action in progress */ + DOOR_OPENING, /* A door is opening */ + DOOR_OPEN, /* A door is open */ + DOOR_CLOSING, /* A door is closing */ +}; + +/**************************************************************************** + * Private Function Prototypes + ****************************************************************************/ + +static void trv_door_startopen(void); +static void trv_door_animation(void); + +/**************************************************************************** + * Public Data + ****************************************************************************/ + +/* Variables to track status of a door */ + +struct trv_opendoor_s g_opendoor; + +/**************************************************************************** + * Private Data + ****************************************************************************/ + +/**************************************************************************** + * Private Functions + ****************************************************************************/ + +/**************************************************************************** + * Name: trv_door_startopen + * + * Description: + * + ***************************************************************************/ + +static void trv_door_startopen (void) +{ + FAR struct trv_rect_data_s *rect; + + /* Check if a door is already opening */ + + if (g_opendoor.state == DOOR_IDLE) + { + /* Test if there is a door within three steps in front of the player */ + /* Try the X planes first */ + + rect = trv_ray_test_xplane(&g_player, 3*STEP_DISTANCE, + g_player.yaw, g_player_height); + + /* If there is no X door in front of the player, then try the Y Planes + * (it is assumed that there would not be doors this close in both + * planes!) + */ + + if (!rect || !IS_DOOR(rect)) + { + rect = trv_ray_test_yplane(&g_player, 3*STEP_DISTANCE, + g_player.yaw, g_player_height); + } + + /* Check if we found a door in either the X or Y plane. */ + + if (rect && IS_DOOR(rect)) + { + /* We now have found a door to open. Set the door open sequence + * in motion + */ + + g_opendoor.rect = rect; + g_opendoor.state = DOOR_OPENING; + g_opendoor.zbottom = rect->vstart; + g_opendoor.zdist = 0; + + /* Mark the door's attribute to indicate that it is in motion */ + + rect->attribute |= MOVING_DOOR_PLANE; + } + } +} + +/**************************************************************************** + * Name: trv_door_animation + * Description: + * This function is called every frame when a door is in motion. + ****************************************************************************/ + +static void trv_door_animation(void) +{ + /* What we do next depends upon the state of the door state machine */ + + switch (g_opendoor.state) + { + /* A door is opening */ + + case DOOR_OPENING : + + /* Raise the door a little */ + + g_opendoor.zbottom += DOOR_ZSTEP; + g_opendoor.zdist += DOOR_ZSTEP; + + /* When the bottom of the door is above the player's head, we will + * say that the door is open + */ + + if (g_opendoor.zbottom > g_player.z) + { + g_opendoor.rect->attribute |= OPEN_DOOR_PLANE; + } + + /* Check if the door is fully open */ + + if (g_opendoor.zbottom >= g_opendoor.rect->vend) + { + /* Make sure that the door does not open wider than it is tall */ + + g_opendoor.zbottom = g_opendoor.rect->vend; + g_opendoor.zdist = g_opendoor.rect->vend - g_opendoor.rect->vstart; + + /* The door is done opening, the next state is the DOOR_OPEN state + * where we will hold the door open a while + */ + + g_opendoor.state = DOOR_OPEN; + + /* Initialize the countdown timer which will determine how long + * the door is held open + */ + + g_opendoor.clock = DOOR_OPEN_WAIT; + } + break; + + /* The door is open */ + + case DOOR_OPEN : + /* Decrement the door open clock. When it goes to zero, it is time + * to begin closing the door + */ + + if (--g_opendoor.clock <= 0) + { + /* The door is done opening, the next state is the DOOR_CLOSING + * states + */ + + g_opendoor.state = DOOR_CLOSING; + + /* Lower the door a little */ + + g_opendoor.zbottom -= DOOR_ZSTEP; + g_opendoor.zdist -= DOOR_ZSTEP; + + /* When the bottom of the door is below the player's head, we + * will say that the door is closed + */ + + if (g_opendoor.zbottom <= g_player.z) + { + g_opendoor.rect->attribute &= ~OPEN_DOOR_PLANE; + } + } + break; + + /* The door is closing */ + + case DOOR_CLOSING : + /* Lower the door a little */ + + g_opendoor.zbottom -= DOOR_ZSTEP; + g_opendoor.zdist -= DOOR_ZSTEP; + + /* When the bottom of the door is below the player's head, we will + * say that the door is closed + */ + + if (g_opendoor.zbottom <= g_player.z) + { + g_opendoor.rect->attribute &= ~OPEN_DOOR_PLANE; + } + + /* Check if the door is fully closed */ + + if (g_opendoor.zdist <= 0) + { + /* Indicate that the door is no longer in motion */ + + g_opendoor.rect->attribute &= ~(OPEN_DOOR_PLANE|MOVING_DOOR_PLANE); + + /* Indicate that the entire door movement sequence is done */ + + g_opendoor.rect = NULL; + g_opendoor.state = DOOR_IDLE; + } + break; + + /* There is nothing going on! */ + + case DOOR_IDLE : + default : + break; + } +} + +/**************************************************************************** + * Public Functions + ****************************************************************************/ + +/**************************************************************************** + * Name: trv_door_initialize + * + * Description: + * This function performs initialization of the door animation logic + * + ****************************************************************************/ + +void trv_door_initialize(void) +{ + /* Initialize the g_opendoor structure */ + + g_opendoor.rect = NULL; + g_opendoor.state = DOOR_IDLE; +} + +/*************************************************************************** + * Name: trv_door_animate + * + * Description: + * This function must be called on each cycle. It checks if the player + * is attempting to open a door. If so, the appropriate door animation + * is started. This function then calls trv_door_animation which must be + * called on each cycle to perform the door movement. + * + ***************************************************************************/ + +void trv_door_animate(void) +{ + /* Check if the user is trying to open a door. */ + + if (g_trv_input.dooropen && g_opendoor.state == DOOR_IDLE) + { + + /* Start the door opening action */ + + trv_door_startopen(); + } + + /* Process any necessary door movement animation on this cycle */ + + if (g_opendoor.state != DOOR_IDLE) + { + /* Perform the animation */ + + trv_door_animation(); + } +} diff --git a/graphics/traveler/src/trv_fsutils.c b/graphics/traveler/src/trv_fsutils.c new file mode 100644 index 000000000..ad0f8d30d --- /dev/null +++ b/graphics/traveler/src/trv_fsutils.c @@ -0,0 +1,97 @@ +/******************************************************************************* + * apps/graphics/traveler/src/trv_fsutils.c + * Miscellaneous file access utilities + * + * Copyright (C) 2014 Gregory Nutt. All rights reserved. + * Author: Gregory Nutt + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in + * the documentation and/or other materials provided with the + * distribution. + * 3. Neither the name NuttX nor the names of its contributors may be + * used to endorse or promote products derived from this software + * without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, + * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, + * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS + * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED + * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN + * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + * + ****************************************************************************/ + +/**************************************************************************** + * Included files + ****************************************************************************/ + +#include "trv_types.h" +#include "trv_fsutils.h" + +#include +#include + +/**************************************************************************** + * Public Functions + ****************************************************************************/ + +/**************************************************************************** + * Name: trv_read_decimal + * + * Description: + * Read a decimal number from the steam 'fp' + * + ***************************************************************************/ + +int16_t trv_read_decimal(FAR FILE *fp) +{ + int16_t value = 0; + bool negative = false; + int ch; + + /* Skip over any leading whitespace */ + + do + { + ch = getc(fp); + } + while (isspace(ch)); + + /* if the first character is '-', then its a negative number */ + + if (ch == '-') + { + negative = true; + ch = getc(fp); + } + + /* Now get the unsigned portion of the number */ + + while (ch >= '0' && ch <= '9') + { + value = 10 *value + (ch - (int)'0'); + ch = getc(fp); + } + + /* Apply the negation, if appropriate */ + + if (negative) + { + value = -value; + } + + return value; +} diff --git a/graphics/traveler/src/trv_graphicfile.c b/graphics/traveler/src/trv_graphicfile.c new file mode 100644 index 000000000..4e598fd58 --- /dev/null +++ b/graphics/traveler/src/trv_graphicfile.c @@ -0,0 +1,167 @@ +/******************************************************************************* + * apps/graphics/traveler/src/trv_graphicfile.c + * Load image from a graphic file + * + * Copyright (C) 2014 Gregory Nutt. All rights reserved. + * Author: Gregory Nutt + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in + * the documentation and/or other materials provided with the + * distribution. + * 3. Neither the name NuttX nor the names of its contributors may be + * used to endorse or promote products derived from this software + * without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, + * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, + * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS + * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED + * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN + * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + * + ****************************************************************************/ + +/**************************************************************************** + * Included files + ****************************************************************************/ + +#include "trv_types.h" +#include "trv_mem.h" +#include "trv_pcx.h" +#include "trv_graphicfile.h" + +#include +#include + +/**************************************************************************** + * Public Functions + ****************************************************************************/ + +/**************************************************************************** + * Name: tvr_graphicfile_read + * + * Description: + * Read a bitmap texture from a graphic file + * + * NOTE: This function exists only as extension pointer where additional + * graphic file formats can be supported. Currently only the ancient + * PCX format is supported. + * + ***************************************************************************/ + +FAR struct trv_graphicfile_s *tvr_graphicfile_read(char *filename) +{ + FAR struct trv_graphicfile_s *gfile = NULL; + FAR FILE *fp; + + /* Open the graphic file for reading */ + + fp = fopen(filename, "rb"); + if (fp == NULL) + { + int errcode = errno; + fprintf(stderr, "ERROR: Failed to open graphic file %s: %d\n", + filename, errcode); + return NULL; + } + + /* Load the graphic file into memory. + * + * REVISIT: Here would be the place where we would decide on the format of + * the graphic file. Here we just assume that the file is PCX. + */ + + gfile = trv_load_pcx(fp, filename); + if (gfile == NULL) + { + fprintf(stderr, "ERROR: Failed to load graphic file %s\n", filename); + } + + fclose(fp); + return gfile; +} + +/************************************************************************* + * Name: trv_graphicfile_new + * + * Description: + * Allocate a new graphic file structure + * + ************************************************************************/ + +FAR struct trv_graphicfile_s *trv_graphicfile_new(void) +{ + FAR struct trv_graphicfile_s *gfile; + + + gfile = (FAR struct trv_graphicfile_s*) + trv_malloc(sizeof(struct trv_graphicfile_s)); + if (gfile) + { + gfile->transparent_entry = -1; + } + + return gfile; +} + +/************************************************************************* + * Name: trv_graphicfile_free + * + * Description: + * Free the graphic file structure after also freeing in additional + * resources attached to the structure. + + ************************************************************************/ + +void trv_graphicfile_free(FAR struct trv_graphicfile_s *gfile) +{ + if (gfile != NULL) + { + if (gfile->palette != NULL) + { + trv_free(gfile->palette); + } + + if (gfile->bitmap != NULL) + { + trv_free(gfile->bitmap); + } + + trv_free(gfile); + } +} + +/************************************************************************* + * Name: trv_graphicfile_pixel + * + * Description: + * Return the RGB color value for a pixel at location (x,y) in the + * texture bitmap. + * + ************************************************************************/ + +struct trv_color_rgb_s +trv_graphicfile_pixel(FAR struct trv_graphicfile_s *gfile, int x, int y) +{ + /* REVISIT: This only works for paletted graphics files like PCX. + * For the paletted color, the color lookup values lies at the (x,y) + * position in the image. For a true color image, the RGB value would + * lie at the (x,y) position, + */ + + int ndx = gfile->bitmap[y * gfile->width + x]; + return gfile->palette[ndx]; +} diff --git a/graphics/traveler/src/trv_graphics.c b/graphics/traveler/src/trv_graphics.c index 02d9b61b9..c37086cab 100644 --- a/graphics/traveler/src/trv_graphics.c +++ b/graphics/traveler/src/trv_graphics.c @@ -41,6 +41,7 @@ #include "trv_main.h" #include "trv_mem.h" #include "trv_color.h" +#include "trv_raycntl.h" #include "trv_debug.h" #include "trv_graphics.h" @@ -130,8 +131,8 @@ static void trv_fb_initialize(FAR struct trv_graphics_info_s *ginfo) trv_abort("getvideoinfo() failed\n"); } - ginfo->hwwidth = vinfo.xres; - ginfo->hwheight = vinfo.yres; + ginfo->xres = vinfo.xres; + ginfo->yres = vinfo.yres; ret = fbdev->getplaneinfo(fbdev, 0, &pinfo); if (ret < 0) @@ -329,24 +330,21 @@ void trv_row_update(struct trv_graphics_info_s *ginfo, FAR const trv_pixel_t *src, FAR dev_pixel_t *dest) { - trv_coord_t hexpand; dev_pixel_t pixel; trv_coord_t srccol; int i; /* Loop for each column in the src render buffer */ - hexpand = (1 << ginfo->hscale); - - for (srccol = 0; srccol < ginfo->swwidth; srccol++) + for (srccol = 0; srccol < TRV_SCREEN_WIDTH; srccol++) { /* Map the source pixel */ pixel = ginfo->palette.lut[*src++]; - /* Copy it to the destination, expanding as necessary */ + /* Expand pixels horizontally via pixel replication */ - for (i = 0; i < hexpand; i++) + for (i = 0; i < ginfo->xscale; i++) { *dest++ = pixel; } @@ -383,8 +381,8 @@ void trv_display_update(struct trv_graphics_info_s *ginfo, int trv_graphics_initialize(FAR struct trv_graphics_info_s *ginfo) { - int swwidth; - int swheight; + int width; + int height; int scale; /* Initialize the graphics device and get information about the display */ @@ -397,31 +395,39 @@ int trv_graphics_initialize(FAR struct trv_graphics_info_s *ginfo) trv_nxsu_initialize(ginfo); #endif + /* Check the size of the display */ + + width = ginfo->xres; + height = ginfo->yres; + + if (width < TRV_SCREEN_WIDTH || height < TRV_SCREEN_HEIGHT) + { + trv_abort("ERROR: Display is too small\n"); + } + /* Check if we need to scale the image */ - swwidth = ginfo->hwwidth; - scale = 0; + scale = 0; - while (swwidth > MAX_REND_WIDTH) + while (width >= TRV_SCREEN_WIDTH) { - swwidth >>= 1; + width -= TRV_SCREEN_WIDTH; scale++; } - ginfo->swwidth = swwidth; - ginfo->hscale = scale; + ginfo->xscale = scale; + ginfo->xoffset = (width >> 1); + ginfo->imgwidth = scale * TRV_SCREEN_WIDTH * sizeof(dev_pixel_t); - swheight = ginfo->hwheight; - scale = 0; - - while (swheight > MAX_REND_WIDTH) + scale = 0; + while (height >= TRV_SCREEN_HEIGHT) { - swheight >>= 1; + height -= TRV_SCREEN_HEIGHT; scale++; } - ginfo->swheight = swheight; - ginfo->vscale = scale; + ginfo->yscale = scale; + ginfo->yoffset = (height >> 1); /* Allocate buffers * @@ -431,7 +437,7 @@ int trv_graphics_initialize(FAR struct trv_graphics_info_s *ginfo) */ ginfo->swbuffer = (trv_pixel_t*) - trv_malloc(swwidth * swheight * sizeof(trv_pixel_t)); + trv_malloc(TRV_SCREEN_WIDTH * TRV_SCREEN_HEIGHT * sizeof(trv_pixel_t)); if (!ginfo->swbuffer) { trv_abort("ERROR: Failed to allocate render buffer\n"); @@ -448,8 +454,7 @@ int trv_graphics_initialize(FAR struct trv_graphics_info_s *ginfo) */ #ifdef CONFIG_NX - ginfo->hwbuffer = (trv_pixel_t*) - trv_malloc(ginfo->hwwidth * sizeof(dev_pixel_t)); + ginfo->hwbuffer = (trv_pixel_t*)trv_malloc(ginfo->imgwidth); if (!ginfo->hwbuffer) { trv_abort("ERROR: Failed to allocate hardware line buffer\n"); @@ -503,28 +508,33 @@ void trv_graphics_terminate(FAR struct trv_graphics_info_s *ginfo) void trv_display_update(struct trv_graphics_info_s *ginfo) { - FAR const uint8_t *src = (FAR const uint8_t *)ginfo->swbuffer; - FAR uint8_t *dest = (FAR uint8_t *)ginfo->hwbuffer; + FAR const uint8_t *src; + FAR uint8_t *dest; trv_coord_t srcrow; #ifdef CONFIG_NX trv_coord_t destrow; #else FAR uint8_t *first; - trv_coord_t lnwidth; #endif - trv_coord_t vexpand; int i; - /* Loop for each row in the srce render buffer */ + /* Get the star tof the first source row */ + + src = (FAR const uint8_t *)ginfo->swbuffer; + + /* Get the start of the first destination row */ + + dest = (FAR uint8_t *)ginfo->hwbuffer + + (ginfo->yoffset * ginfo->stride) + + (ginfo->xoffset * sizeof(dev_pixel_t)); + + /* Loop for each row in the src render buffer */ - vexpand = (1 << ginfo->vscale); #ifdef CONFIG_NX destrow = 0; -#else - lnwidth = ginfo->hwwidth * sizeof(dev_pixel_t); #endif - for (srcrow = 0; srcrow < ginfo->swheight; srcrow++) + for (srcrow = 0; srcrow < TRV_SCREEN_HEIGHT; srcrow++) { /* Transfer the row to the device row/buffer */ @@ -543,7 +553,7 @@ void trv_display_update(struct trv_graphics_info_s *ginfo) /* Then replicate as many times as is necessary */ - for (i = 1; i < vexpand; i++) + for (i = 1; i < ginfo->yscale; i++) { #ifdef CONFIG_NX /* Transfer the row buffer to the NX window */ @@ -553,14 +563,14 @@ void trv_display_update(struct trv_graphics_info_s *ginfo) #else /* Point to the next row in the frame buffer */ - memcpy(dest, first, lnwidth); + memcpy(dest, first, ginfo->imgwidth); dest += ginfo->stride; #endif } /* Point to the next src row */ - src += ginfo->swwidth; + src += TRV_SCREEN_WIDTH; } } diff --git a/graphics/traveler/src/trv_input.c b/graphics/traveler/src/trv_input.c index ea100de5c..44644e150 100644 --- a/graphics/traveler/src/trv_input.c +++ b/graphics/traveler/src/trv_input.c @@ -1,6 +1,5 @@ /**************************************************************************** * apps/graphics/traveler/trv_input.c - * This file contains the main logic for the NuttX version of Traveler * * Copyright (C) 2014 Gregory Nutt. All rights reserved. * Author: Gregory Nutt diff --git a/graphics/traveler/src/trv_main.c b/graphics/traveler/src/trv_main.c index 594513a9a..029fd4065 100644 --- a/graphics/traveler/src/trv_main.c +++ b/graphics/traveler/src/trv_main.c @@ -219,7 +219,7 @@ int traveler_main(int argc, char *argv[]) ret = trv_world_create(world_filename); if (ret < 0) { - trv_abort("ERROR: %d loading world file %s: %d\n", + trv_abort("ERROR: Failed to load world file %s: %d\n", world_filename, ret); } @@ -257,11 +257,11 @@ int traveler_main(int argc, char *argv[]) /* Paint the back drop */ - trv_rend_backdrop(&g_trv_player); + trv_rend_backdrop(&g_player, &g_trv_ginfo); /* Render the 3-D view */ - trv_raycaster(&g_trv_player, &g_trv_ginfo); + trv_raycaster(&g_player, &g_trv_ginfo); /* Display the world. */ diff --git a/graphics/traveler/src/trv_mem.c b/graphics/traveler/src/trv_mem.c index 84e85c3d0..816e8081d 100644 --- a/graphics/traveler/src/trv_mem.c +++ b/graphics/traveler/src/trv_mem.c @@ -1,5 +1,6 @@ /**************************************************************************** * apps/graphics/traveler/src/trv_mem.c + * Memory management support * * Copyright (C) 2014 Gregory Nutt. All rights reserved. * Author: Gregory Nutt diff --git a/graphics/traveler/src/trv_nxbkgd.c b/graphics/traveler/src/trv_nxbkgd.c index 08a54a834..e0f34a512 100644 --- a/graphics/traveler/src/trv_nxbkgd.c +++ b/graphics/traveler/src/trv_nxbkgd.c @@ -1,5 +1,6 @@ /**************************************************************************** * apps/graphics/traveler/trv_nxbkgd.c + * NX background window callback handlers * * Copyright (C) 2014 Gregory Nutt. All rights reserved. * Author: Gregory Nutt diff --git a/graphics/traveler/src/trv_nxserver.c b/graphics/traveler/src/trv_nxserver.c index e2cedaa28..be753299d 100644 --- a/graphics/traveler/src/trv_nxserver.c +++ b/graphics/traveler/src/trv_nxserver.c @@ -1,5 +1,6 @@ /**************************************************************************** - * apps/graphics/traveler/trv_nxlistener.c + * apps/graphics/traveler/trv_nxserver.c + * NX server logic * * Copyright (C) 2014 Gregory Nutt. All rights reserved. * Author: Gregory Nutt diff --git a/graphics/traveler/src/trv_paltbl.c b/graphics/traveler/src/trv_paltbl.c new file mode 100644 index 000000000..47cb82237 --- /dev/null +++ b/graphics/traveler/src/trv_paltbl.c @@ -0,0 +1,348 @@ +/******************************************************************************* + * apps/graphics/traveler/src/trv_paltbl.c + * This file contains the logic that creates the range palette table that is + * used to modify the palette with range to hit + * + * Copyright (C) 2014 Gregory Nutt. All rights reserved. + * Author: Gregory Nutt + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in + * the documentation and/or other materials provided with the + * distribution. + * 3. Neither the name NuttX nor the names of its contributors may be + * used to endorse or promote products derived from this software + * without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, + * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, + * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS + * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED + * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN + * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + * + ****************************************************************************/ + +/**************************************************************************** + * Included files + ****************************************************************************/ + +#include "trv_types.h" +#include "trv_mem.h" +#include "trv_color.h" +#include "trv_fsutils.h" +#include "trv_world.h" +#include "trv_bitmaps.h" +#include "trv_paltable.h" + +#include +#include + +/**************************************************************************** + * Pre-processor Definitions + ****************************************************************************/ + +/* This indicates the maximum number of palette ranges which may be + * specified in the file + */ + +#ifdef CONFIG_GRAPHICS_TRAVELER_PALRANGES +# define MAX_PAL_RANGES 64 +#endif + +/**************************************************************************** + * Private Type Declarations + ***************************************************************************/ + +#ifdef CONFIG_GRAPHICS_TRAVELER_PALRANGES +struct trv_palrange_s +{ + uint8_t firstcolor; + int8_t colorrange; + uint8_t clipColor; +}; +#endif + +/**************************************************************************** + * Public Data + ****************************************************************************/ + +/* This is the palette table which is used to adjust the texture values + * with distance + */ + +trv_pixel_t *g_paltable[NUM_ZONES]; + +/**************************************************************************** + * Private Functions + ****************************************************************************/ + +/**************************************************************************** + * Name: trv_allocate_paltbl + * + * Description: + * + ***************************************************************************/ + +static int trv_allocate_paltbl(uint32_t entrysize) +{ + int i; + + for (i = 0; i < NUM_ZONES; i++) + { + g_paltable[i] = (FAR trv_pixel_t*) + trv_malloc(entrysize * sizeof(trv_pixel_t)); + + if (g_paltable[i] == NULL) + { + return -ENOMEM; + } + } + + return OK; +} + +/**************************************************************************** + * Public Functions + ****************************************************************************/ + +/**************************************************************************** + * Name: trv_load_paltable + * + * Description: + * This function loads the g_paltable from the specified file + * + ***************************************************************************/ + +int trv_load_paltable(FAR const char *file) +{ +#ifdef CONFIG_GRAPHICS_TRAVELER_PALRANGES + struct trv_palrange_s ranges[MAX_PAL_RANGES]; + trv_pixel_t plotcolor; + FAR trv_pixel_t *palptr; + FAR FILE *fp; + int16_t nranges; + int16_t zone; + int16_t palndx; + int ret; + int i; + + /* Open the file which contains the palette table */ + + fp = fopen(file, "r"); + if (!fp) + { + int errcode = errno; + fprintf(stderr, "ERROR: Failed to open %s: %d\n", file, errcode); + return -errcode; + } + + /* Read the number of ranges from the file */ + + nranges = trv_read_decimal(fp); + if (nranges > MAX_PAL_RANGES) + { + fprintf(stderr, "ERROR: Too many ranges: %d\n", nranges); + fclose(fp); + return -E2BIG; + } + + /* Then read all of the palette range data from the file */ + + for (i = 0; i < nranges; i++) + { + ranges[i].firstcolor = trv_read_decimal(fp); + ranges[i].colorrange = trv_read_decimal(fp); + ranges[i].clipColor = trv_read_decimal(fp); + } + + /* We are now done with the input file */ + + fclose(fp); + + /* Allocate memory to hold the palette range mapping data */ + + ret = trv_allocate_paltbl(PALETTE_SIZE); + if (ret < 0) + { + fprintf(stderr, + "ERROR: Failed to allocate range mapping table: %d\n", + ret); + return ret; + } + + /* Process each distance zone */ + + for (zone = 0; zone < NUM_ZONES; zone++) + { + /* Point to the palette mapping data for this zone */ + + palptr = g_paltable[zone]; + + /* Process each possible palette index within the zone */ + + for (palndx = 0; palndx < PALETTE_SIZE; palndx++) + { + /* Assume that the range will not be found. In this case, we + * will perform the 1-to-1 mapping + */ + + plotcolor = palndx; + + /* Find the range the color is in. */ + + for (i = 0; i < nranges; i++) + { + /* Check if the color range is ascending or descending */ + + if (ranges[i].colorrange < 0) + { + /* The colors are descending */ + + if ((palndx <= ranges[i].firstcolor) && + (palndx > (ranges[i].firstcolor + ranges[i].colorrange))) + { + /* Found it, set the new plotcolor */ + + if (plotcolor > zone) + { + /* Offset the color by the zone we are processing */ + + plotcolor -= zone; + + /* Check if we have exceeded the range of this + * color. If so, then set the color to the + * clipColor + */ + + if (plotcolor <= ranges[i].firstcolor + ranges[i].colorrange) + { + plotcolor = ranges[i].clipColor; + } + } + else + { + plotcolor = ranges[i].clipColor; + } + + /* Now break out of the loop */ + + break; + } + } + + /* The colors are ascending */ + + else if ((palndx >= ranges[i].firstcolor) && + (palndx < (ranges[i].firstcolor + ranges[i].colorrange))) + { + /* Found it, set the new plotcolor */ + + plotcolor += zone; + + /* Check if we have exceeded the range of this color. If so, + * then set the color to black + */ + + if (plotcolor >= ranges[i].firstcolor + ranges[i].colorrange) + { + plotcolor = ranges[i].clipColor; + } + + /* Now break out of the loop */ + + break; + } + } + + /* Save the plotcolor in the g_paltable */ + + palptr[palndx] = plotcolor; + + } + } + + return OK; +#else + FILE *fp; + int16_t zone; + int16_t palndx; + FAR uint8_t *palptr; + int ret; + + /* Allocate memory to hold the palette range mapping data */ + + ret = trv_allocate_paltbl(PALETTE_SIZE); + if (ret < 0) + { + fprintf(stderr, + "ERROR: Failed to allocate range mapping table: %d\n", + ret); + return ret; + } + + /* Open the file which contains the palette table */ + + fp = fopen(file, "r"); + if (!fp) + { + int errcode = errno; + fprintf(stderr, "ERROR: Failed to open %s: %d\n", file, errcode); + return -errcode; + } + + /* Process each distance zone */ + + for (zone = 0; zone < NUM_ZONES; zone++) + { + /* Point to the palette mapping data for this zone */ + + palptr = GET_PALPTR(zone); + + /* Process each possible palette index within the zone */ + + for (palndx = 0; palndx < PALETTE_SIZE; palndx++) + { + /* Read the data into g_paltable */ + + palptr[palndx] = trv_read_decimal(fp); + } + } + + fclose(fp); + return OK; +#endif /* CONFIG_GRAPHICS_TRAVELER_PALRANGES */ +} + +/************************************************************************* + * Function: trv_release_paltable + * + * Description: + * + ************************************************************************/ + +void trv_release_paltable(void) +{ + int i; + + for (i = 0; i < NUM_ZONES; i++) + { + if (g_paltable[i]) + { + trv_free(g_paltable[i]); + g_paltable[i] = NULL; + } + } +} diff --git a/graphics/traveler/src/trv_pcx.c b/graphics/traveler/src/trv_pcx.c new file mode 100644 index 000000000..5ff2d0cb7 --- /dev/null +++ b/graphics/traveler/src/trv_pcx.c @@ -0,0 +1,217 @@ +/******************************************************************************* + * apps/graphics/traveler/src/trv_pcx.c + * PCX graphic file support + * + * Copyright (C) 2014 Gregory Nutt. All rights reserved. + * Author: Gregory Nutt + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in + * the documentation and/or other materials provided with the + * distribution. + * 3. Neither the name NuttX nor the names of its contributors may be + * used to endorse or promote products derived from this software + * without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, + * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, + * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS + * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED + * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN + * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + * + ****************************************************************************/ + +/**************************************************************************** + * Included files + ****************************************************************************/ + +#include "trv_types.h" +#include "trv_mem.h" +#include "trv_color.h" +#include "trv_bitmaps.h" +#include "trv_graphicfile.h" +#include "trv_pcx.h" + +#include + +/**************************************************************************** + * Private Functions + ****************************************************************************/ + +/**************************************************************************** + * Name: trv_load_pcxheader + * + * Description: + * + ***************************************************************************/ + +static void trv_load_pcxheader(FAR FILE *fp, struct pcx_header_s *header) +{ + FAR uint8_t *tmpbuffer; + int i; + + /* Load the header */ + + tmpbuffer = (FAR uint8_t*)header; + + for (i = 0; i < SIZEOF_PCX_HEADER; i++) + { + tmpbuffer[i] = getc(fp); + } +} + +/**************************************************************************** + * Name: trv_load_pcxdata + * + * Description: + * + ***************************************************************************/ + +static void trv_load_pcxdata(FAR FILE *fp, int32_t imagesize, + FAR uint8_t *imagebuffer) +{ + uint32_t count; + int16_t nbytes; + uint8_t data; + + count = 0; + while(count <= imagesize) + { + /* Get the first piece of data */ + + data = getc(fp); + + /* Is this a run length encoding? */ + + if ((data >= 192) /* && (data <= 255) */) + { + /* How many bytes in run? */ + + nbytes = data - 192; + + /* Get the actual data for the run */ + + data = getc(fp); + + /* Replicate data in buffer nbytes times */ + + while(nbytes-- > 0) + { + imagebuffer[count++] = data; + } + } + else + { + /* Actual data, just copy it into buffer at next location */ + + imagebuffer[count++] = data; + } + } +} + +/**************************************************************************** + * Name: trv_load_pcxpalette + * + * Description: + * + ***************************************************************************/ + +static void trv_load_pcxpalette(FAR FILE *fp, + FAR struct trv_color_rgb_s *palette) +{ + int i; + + /* Move to end of file then back up 768 bytes i.e. to beginning of palette */ + + (void)fseek(fp, -768L, SEEK_END); + + /* Load the PCX palette into the palette structure */ + + for (i = 0; i < 256; i++) + { + /* Get the RGB components */ + + palette[i].red = getc(fp); + palette[i].green = getc(fp); + palette[i].blue = getc(fp); + } +} + +/**************************************************************************** + * Public Functions + ****************************************************************************/ + +/**************************************************************************** + * Name: trv_load_pcx + * + * Description: + * This function loads a PCX file into a memory. + * + ***************************************************************************/ + +FAR struct trv_graphicfile_s *trv_load_pcx(FAR FILE *fp, + FAR const char *filename) +{ + struct pcx_header_s header; + FAR trv_pixel_t *buffer; + FAR struct trv_graphicfile_s *gfile; + FAR struct trv_color_rgb_s *palette; + uint16_t imagewidth; + uint16_t imageheight; + uint32_t imagesize; + + /* Load the PCX Header */ + + trv_load_pcxheader(fp, &header); + + /* Allocate Space to hold the image data */ + + imagewidth = header.width - header.x + 1; + imageheight = header.height - header.y + 1; + imagesize = imageheight * imagewidth * sizeof(trv_pixel_t); + buffer = (FAR trv_pixel_t*)trv_malloc(imagesize + 1); + + /* Load the PCX data into the buffer */ + + trv_load_pcxdata(fp, imagesize, (FAR uint8_t*)buffer); + + /* Allocate space to hold the PCX palette */ + + palette = (struct trv_color_rgb_s*) + trv_malloc(sizeof(struct trv_color_rgb_s) * 256); + + gfile = NULL; + if (palette) + { + /* Load the PCX palette */ + + trv_load_pcxpalette(fp, palette); + + /* Now save the resulting data in a struct trv_graphicfile_s structure */ + + gfile = trv_graphicfile_new(); + if (gfile) + { + gfile->palette = palette; + gfile->width = imagewidth; + gfile->height = imageheight; + gfile->bitmap = buffer; + } + } + + return gfile; +} + diff --git a/graphics/traveler/src/trv_planefiles.c b/graphics/traveler/src/trv_planefiles.c new file mode 100644 index 000000000..828d6c06d --- /dev/null +++ b/graphics/traveler/src/trv_planefiles.c @@ -0,0 +1,183 @@ +/******************************************************************************* + * apps/graphics/traveler/src/trv_planefiles.c + * This file contains the logic to manage the world data files + * + * Copyright (C) 2014 Gregory Nutt. All rights reserved. + * Author: Gregory Nutt + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in + * the documentation and/or other materials provided with the + * distribution. + * 3. Neither the name NuttX nor the names of its contributors may be + * used to endorse or promote products derived from this software + * without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, + * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, + * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS + * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED + * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN + * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + * + ****************************************************************************/ + +/**************************************************************************** + * Included files + ****************************************************************************/ + +#include "trv_types.h" +#include "trv_plane.h" + +#include +#include + +/**************************************************************************** + * Pre-processor Definitions + ****************************************************************************/ + +/**************************************************************************** + * Private Type Declarations + ***************************************************************************/ + +/**************************************************************************** + * Private Functions + ****************************************************************************/ + +/**************************************************************************** + * Name: trv_load_worldplane + * + * Description: + * + * This function loads the world data for one plane + * + ***************************************************************************/ + +static int trv_load_worldplane(FAR FILE *fp, FAR struct trv_rect_head_s *head, + uint8_t nrects) +{ + FAR struct trv_rect_list_s *rect; + int ret; + int i; + + for (i = 0; i < nrects; i++) + { + /* Allocate space for the next rectangle */ + + rect = trv_new_plane(); + if (!rect) + { + return -ENOMEM; + } + + /* Read the next rectangle from the input file */ + + ret = fread((char*)&rect->d, RESIZEOF_TRVRECTDATA_T, 1, fp); + if (ret != 1) + { + int errcode = errno; + fprintf(stderr, "Error: read of rectangle %d (of %d) failed\n", + i, nrects); + fprintf(stderr, "feof=%d ferror=%d errno=%d\n", + feof(fp), ferror(fp), errcode); + return -errcode; + } + + /* Put the new rectangle into the plane list */ + + trv_add_plane(rect, head); + } + + return OK; +} + +/**************************************************************************** + * Function: trv_load_planes + * Description: + * This function loads the world data from the opened file + ***************************************************************************/ + +static int trv_load_planes(FAR FILE *fp) +{ + struct trv_planefile_header_s header; + int ret; + + /* Read the plane file header */ + + ret = fread((char*)&header, SIZEOF_TRVPLANEFILEHEADER_T, 1, fp); + if (ret != 1) + { + int errcode = errno; + fprintf(stderr, "Error: Failed to read of file header\n"); + fprintf(stderr, "feof=%d ferror=%d errno=%d\n", + feof(fp), ferror(fp), errcode); + return -errcode; + } + + /* Then load each grid, performing run length (rle) decoding */ + + ret = trv_load_worldplane(fp, &g_xplane, header.nxrects); + if (ret == OK) + { + ret = trv_load_worldplane(fp, &g_yplane, header.nyrects); + } + + if (ret == OK) + { + ret = trv_load_worldplane(fp, &g_zplane, header.nzrects); + } + + return ret; +} + +/**************************************************************************** + * Public Functions + ****************************************************************************/ + +/**************************************************************************** + * Name: trv_load_planefile + * + * Description: + * + * This function opens the input file and loads the world plane data from it + * + ***************************************************************************/ + +int trv_load_planefile(FAR const char *wldfile) +{ + FAR FILE *fp; + int ret; + + /* Open the map file which contains the description of the world */ + + fp = fopen(wldfile, "rb"); + if (fp == NULL) + { + int errcode = errno; + fprintf(stderr, "ERROR: Could not open plane file=\"%s\": %d\n", + wldfile, errcode); + return -errcode; + } + + /* Load the world data from the file */ + + ret = trv_load_planes(fp); + + /* Close the file */ + + fclose(fp); + return ret; +} + diff --git a/graphics/traveler/src/trv_planelists.c b/graphics/traveler/src/trv_planelists.c new file mode 100644 index 000000000..49d5f8e7f --- /dev/null +++ b/graphics/traveler/src/trv_planelists.c @@ -0,0 +1,378 @@ +/******************************************************************************* + * apps/graphics/traveler/src/trv_planelist.c + * This file contains the logic to manage world plane lists. + * + * Copyright (C) 2014 Gregory Nutt. All rights reserved. + * Author: Gregory Nutt + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in + * the documentation and/or other materials provided with the + * distribution. + * 3. Neither the name NuttX nor the names of its contributors may be + * used to endorse or promote products derived from this software + * without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, + * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, + * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS + * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED + * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN + * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + * + ****************************************************************************/ + +/**************************************************************************** + * Included files + ****************************************************************************/ + +#include "trv_types.h" +#include "trv_mem.h" +#include "trv_plane.h" + +/**************************************************************************** + * Public Data + ****************************************************************************/ + +/* The is the world!!! The world is described by lists of rectangles, one + * for each of the X, Y, and Z planes. + */ + +struct trv_rect_head_s g_xplane; /* list of X=plane rectangles */ +struct trv_rect_head_s g_yplane; /* list of Y=plane rectangles */ +struct trv_rect_head_s g_zplane; /* list of Z=plane rectangles */ + +/* "Deallocated" planes are retained in a free list */ + +FAR struct trv_rect_list_s *g_rect_freelist; + +/**************************************************************************** + * Private Functions + ****************************************************************************/ + +/**************************************************************************** + * Name: trv_release_worldplane + * + * Description: + * This function deallocates one plane of the world + * + ***************************************************************************/ + +static void trv_release_worldplane(FAR struct trv_rect_list_s *rect) +{ + FAR struct trv_rect_list_s *next; + + while (rect) + { + next = rect->flink; + trv_free((void *) rect); + rect = next; + } +} + +/**************************************************************************** + * Public Functions + ****************************************************************************/ + +/**************************************************************************** + * Name: trv_initialize_planes + * + * Description: + * + ***************************************************************************/ + +int trv_initialize_planes(void) +{ + g_xplane.head = NULL; + g_xplane.tail = NULL; + g_yplane.head = NULL; + g_yplane.tail = NULL; + g_zplane.head = NULL; + g_zplane.tail = NULL; + g_rect_freelist = NULL; + + return OK; +} + +/**************************************************************************** + * Name: trv_add_plane + * + * Description: + * This function adds a plane to a world plane list + * + ***************************************************************************/ + +void trv_add_plane(FAR struct trv_rect_list_s *rect, + FAR struct trv_rect_head_s *list) +{ + struct trv_rect_list_s *next; + struct trv_rect_list_s *prev; + + /* Search the list to find the location to insert the new rectangle. Each + * is list is maintained in ascending plane order. + */ + + for (next = list->head; + ((next) && (next->d.plane < rect->d.plane)); + next = next->flink); + + /* Add the 'rect' to the spot found in the list. Check if the 'rect' lies + * at the end of the list. + */ + + if (!next) + { + /* No rectangle with plane larger than the one to be added was found + * in the list. The 'rect' goes at the end of the list. + */ + + prev = list->tail; + if (!prev) + { + /* Special case: The list is empty */ + + rect->flink = NULL; + rect->blink = NULL; + list->head = rect; + list->tail = rect; + } + else + { + rect->flink = NULL; + rect->blink = prev; + prev->flink = rect; + list->tail = rect; + } + } + else + { + /* The 'rect' goes just before 'next' */ + + prev = next->blink; + if (!prev) + { + /* Special case: Insert at the head of the list */ + + rect->flink = next; + rect->blink = NULL; + next->blink = rect; + list->head = rect; + } + else + { + /* Insert in the middle of the list */ + + rect->flink = next; + rect->blink = prev; + prev->flink = rect; + next->blink = rect; + } + } +} + +/**************************************************************************** + * Name: trv_move_plane + * + * Description: + * + * This function removes the specified plane from the world plane srclist + * then adds it to the world plane destlist + * + ***************************************************************************/ + +void trv_move_plane(FAR struct trv_rect_list_s *rect, + FAR struct trv_rect_head_s *destlist, + FAR struct trv_rect_head_s *srclist) +{ + /* Un-hook the backward link to the rect */ + + if (rect->flink) + { + rect->flink->blink = rect->blink; + } + else + { + srclist->tail = rect->blink; + } + + /* Un-hook the forward link to the rect */ + + if (rect->blink) + { + rect->blink->flink = rect->flink; + } + else + { + srclist->head = rect->flink; + } + + /* Then add the rect to the specified list */ + + trv_add_plane(rect, destlist); +} + +/**************************************************************************** + * Name: trv_merge_planelists + * + * Description: + * This function concatenates two world plane lists + * + ***************************************************************************/ + +void trv_merge_planelists(FAR struct trv_rect_head_s *outlist, + FAR struct trv_rect_head_s *inlist) +{ + struct trv_rect_list_s *inrect; + struct trv_rect_list_s *nextin; + struct trv_rect_list_s *outrect; + struct trv_rect_list_s *prevout; + + /* Initialize the inner plane search loop */ + + outrect = outlist->head; + + /* Process every plane in the input plane list */ + + for (inrect = inlist->head; (inrect); inrect = nextin) + { + nextin = inrect->flink; + + /* Search the output plane list to find the location to insert the + * input rectangle. Each is list is maintained in ascending plane + * order. + */ + + for (; + outrect && outrect->d.plane < inrect->d.plane; + outrect = outrect->flink); + + /* Add the 'inrect' to the spot found in the list. Check if the + * 'inrect' goes at the one of the ends of the list. + */ + + if (!outrect) + { + /* No rectangle with plane larger than the one to be added + * was found in the list. The 'inrect' goes at the end of + * the list. + */ + + prevout = outlist->tail; + if (!prevout) + { + /* Special case: The list is empty */ + + inrect->flink = NULL; + inrect->blink = NULL; + outlist->head = inrect; + outlist->tail = inrect; + } + else + { + inrect->flink = NULL; + inrect->blink = prevout; + prevout->flink = inrect; + outlist->tail = inrect; + } + } + else + { + /* The 'inrect' goes just before 'outrect' */ + + prevout = outrect->blink; + if (!prevout) + { + /* Special case: Insert at the head of the list */ + + inrect->flink = outrect; + inrect->blink = NULL; + outrect->blink = inrect; + outlist->head = inrect; + } + else + { + /* Insert in the middle of the list */ + + inrect->flink = outrect; + inrect->blink = prevout; + prevout->flink = inrect; + outrect->blink = inrect; + } + } + + /* Set up for the next time through */ + + outrect = inrect; + } + + /* Mark the input list empty */ + + inlist->head = NULL; + inlist->tail = NULL; +} + +/************************************************************************* + * Function: trv_new_plane + * + * Description: + * This function allocates memory for a new plane rectangle. + * + ************************************************************************/ + +FAR struct trv_rect_list_s *trv_new_plane(void) +{ + FAR struct trv_rect_list_s *rect; + + /* Try to get the new structure from the free list */ + + rect = g_rect_freelist; + if (rect) + { + /* Got get... remove it from the g_rect_freelist; */ + + g_rect_freelist = rect->flink; + } + else + { + /* Nothing on the free list. Allocate a new one */ + + rect = (FAR struct trv_rect_list_s*)trv_malloc(sizeof(struct trv_rect_list_s)); + } + + return rect; +} + +/************************************************************************* + * Name: trv_release_planes + * + * Description: + * + * This function deallocates the entire world. + * + ************************************************************************/ + +void trv_release_planes(void) +{ + /* Release all world planes */ + + trv_release_worldplane(g_xplane.head); + trv_release_worldplane(g_yplane.head); + trv_release_worldplane(g_zplane.head); + trv_release_worldplane(g_rect_freelist); + + /* Re-initialize the world plane lists */ + + trv_initialize_planes(); +} diff --git a/graphics/traveler/src/trv_pov.c b/graphics/traveler/src/trv_pov.c index 33d66a1f6..7472d72c9 100644 --- a/graphics/traveler/src/trv_pov.c +++ b/graphics/traveler/src/trv_pov.c @@ -1,5 +1,6 @@ /**************************************************************************** * apps/graphics/traveler/trv_pov.c + * This file contains the logic which manages player's point-of-view (POV) * * Copyright (C) 2014 Gregory Nutt. All rights reserved. * Author: Gregory Nutt @@ -53,7 +54,7 @@ /* This structure defines the current camera position of the player's eyes */ -struct trv_camera_s g_trv_player; +struct trv_camera_s g_player; /**************************************************************************** * Private Data @@ -77,17 +78,17 @@ static void trv_new_viewangle(void) { /* Determine the amount to rotate on this cycle */ - g_trv_player.yaw += g_trv_input.yawrate; + g_player.yaw += g_trv_input.yawrate; /* Make sure that yaw is still within range */ - if (g_trv_player.yaw >= ANGLE_360) + if (g_player.yaw >= ANGLE_360) { - g_trv_player.yaw -= ANGLE_360; + g_player.yaw -= ANGLE_360; } - else if (g_trv_player.yaw < ANGLE_0) + else if (g_player.yaw < ANGLE_0) { - g_trv_player.yaw += ANGLE_360; + g_player.yaw += ANGLE_360; } } @@ -95,33 +96,33 @@ static void trv_new_viewangle(void) if (g_trv_input.pitchrate != 0) { - g_trv_player.pitch += g_trv_input.pitchrate; + g_player.pitch += g_trv_input.pitchrate; /* Make sure that pitch is still within range */ - if (g_trv_player.pitch >= ANGLE_360) + if (g_player.pitch >= ANGLE_360) { - g_trv_player.pitch -= ANGLE_360; + g_player.pitch -= ANGLE_360; } - else if (g_trv_player.pitch < ANGLE_0) + else if (g_player.pitch < ANGLE_0) { - g_trv_player.pitch += ANGLE_360; + g_player.pitch += ANGLE_360; } /* Don't let the player look up more than thirty degrees */ - if (g_trv_player.pitch > ANGLE_30 && - g_trv_player.pitch < ANGLE_180) + if (g_player.pitch > ANGLE_30 && + g_player.pitch < ANGLE_180) { - g_trv_player.pitch = ANGLE_30; + g_player.pitch = ANGLE_30; } /* Don't let the player look down more than thirty degrees */ - else if (g_trv_player.pitch < (ANGLE_360 - ANGLE_30) && - g_trv_player.pitch > ANGLE_180) + else if (g_player.pitch < (ANGLE_360 - ANGLE_30) && + g_player.pitch > ANGLE_180) { - g_trv_player.pitch = (ANGLE_360 - ANGLE_30); + g_player.pitch = (ANGLE_360 - ANGLE_30); } } } @@ -144,7 +145,7 @@ static void trv_new_playerposition(void) /* Assume that we are moving forward */ - move_angle = g_trv_player.yaw; + move_angle = g_player.yaw; left_angle = ANGLE_90; fwd_distance = g_trv_input.fwdrate; left_distance = g_trv_input.leftrate; @@ -179,12 +180,12 @@ static void trv_new_playerposition(void) if (left_distance > 0) { - move_angle = g_trv_player.yaw - left_angle; + move_angle = g_player.yaw - left_angle; } else if (left_distance < 0) { left_distance = -left_distance; - move_angle = g_trv_player.yaw + left_angle; + move_angle = g_player.yaw + left_angle; } if (move_angle < ANGLE_0) @@ -211,18 +212,18 @@ static void trv_new_playerposition(void) * and clip these components to avoid collisions with walls and objects */ - g_trv_player.x += - trv_rayclip_player_xmotion(&g_trv_player, move_distance, + g_player.x += + trv_rayclip_player_xmotion(&g_player, move_distance, move_angle, g_trv_input.stepheight); - g_trv_player.y += - trv_rayclip_player_ymotion(&g_trv_player, move_distance, + g_player.y += + trv_rayclip_player_ymotion(&g_player, move_distance, move_angle, g_trv_input.stepheight); /* Adjust the player's vertical position (he may have fallen down or * stepped up something. */ - g_trv_player.z += trv_ray_adjust_zpos(&g_trv_player, g_player_height); + g_player.z += trv_ray_adjust_zpos(&g_player, g_player_height); } /**************************************************************************** @@ -239,7 +240,7 @@ static void trv_new_playerposition(void) void trv_pov_reset(void) { - g_trv_player = g_initial_camera; + g_player = g_initial_camera; } /**************************************************************************** diff --git a/graphics/traveler/src/trv_rayavoid.c b/graphics/traveler/src/trv_rayavoid.c new file mode 100644 index 000000000..cfdccf828 --- /dev/null +++ b/graphics/traveler/src/trv_rayavoid.c @@ -0,0 +1,624 @@ +/******************************************************************************* + * apps/graphics/traveler/src/trv_rayavoid.c + * This file contains the logic which determines if the desired player motion + * would cause a collision with various walls or if the motion would cause + * the player to change his vertical position in the world. This collision + * avoidance logic is also used to determine if there is a door in front of + * the player. + * + * Copyright (C) 2014 Gregory Nutt. All rights reserved. + * Author: Gregory Nutt + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in + * the documentation and/or other materials provided with the + * distribution. + * 3. Neither the name NuttX nor the names of its contributors may be + * used to endorse or promote products derived from this software + * without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, + * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, + * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS + * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED + * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN + * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + * + ****************************************************************************/ + +/***************************************************************************** + * Included files + ****************************************************************************/ + +#include "trv_types.h" +#include "trv_trigtbl.h" +#include "trv_plane.h" +#include "trv_world.h" +#include "trv_rayavoid.h" + +/***************************************************************************** + * Pre-processor Definitions + ****************************************************************************/ + +/* This is the closest that we will allow the player to approach a wall */ + +#define MIN_APPROACH_DISTANCE (64/4) /* One quarter cell */ + +/***************************************************************************** + * Private Data + ****************************************************************************/ + +/* The rayClipPlayX/YMotion functions have the side effect of saving the + * pointer to the wall with which the player collided in the following + * private variable. This gives a "back door" mechanism which is used + * to handle door processing. + */ + +static struct trv_rect_data_s *g_clip_rect; + +/***************************************************************************** + * Public Functions + ****************************************************************************/ + +/***************************************************************************** + * Name: trv_ray_test_xplane + * + * Description: + * + * This function tests if there is an X plane within the supplied distance + * from the player at the supplied angle. This function is used to + * determine in there is a door in front of the player. NOTE: This + * function exploits the normal collision detection logic in + * trv_rayclip_player_xmotion and depends on the side-effect setting of + * g_clip_rect. + * + *****************************************************************************/ + +FAR struct trv_rect_data_s *trv_ray_test_xplane(FAR struct trv_camera_s *pov, + trv_coord_t dist, int16_t yaw, + trv_coord_t height) +{ + (void) trv_rayclip_player_xmotion(pov, dist, yaw, height); + return g_clip_rect; +} + +/***************************************************************************** + * Name: trv_rayclip_player_xmotion + * + * Description: + * This function calculates the acceptable change in the player's position + * along the X component of the specified yaw angle which would not cause + * a collision with an X plane. This logic is essentially a modified X + * ray cast. + * + *****************************************************************************/ + +trv_coord_t trv_rayclip_player_xmotion(FAR struct trv_camera_s *pov, + trv_coord_t dist, int16_t yaw, + trv_coord_t height) +{ + FAR struct trv_rect_list_s *list; /* Points to the current X plane rectangle */ + FAR struct trv_rect_data_s *rect; /* Points to the rectangle data */ + trv_coord_t reqdeltax; + trv_coord_t standoff; + trv_coord_t footlevel; + trv_coord_t relx; /* Relative position of the X plane */ + trv_coord_t absy; /* Absolute Y position at relx given yaw */ + trv_coord_t lastrelx = -1; /* Last relative X position processed */ + + /* Decompose the desired move distance into its X component */ + + reqdeltax = tTOs(g_cos_table[ yaw ] * dist); + + /* Assume that no clipping will be performed */ + + g_clip_rect = NULL; + + /* Perform X raycasting based on the quadrant of the yaw angle */ + /* The first and fourth quadrants correspond to the positive X + * direction (excluding angles 90 and 270). + */ + + if (yaw < ANGLE_90 || yaw > ANGLE_270) + { + /* Calculate the requested distance along the (positive) X axis (adding + * a little to how close the play can come to a wall + */ + + standoff = reqdeltax + MIN_APPROACH_DISTANCE; + + /* This is the position of the player's feet */ + + footlevel = pov->z - height; + + /* Look at every rectangle lying in the X plane */ + + for (list = g_xplane.head; list; list = list->flink) + { + rect = &list->d; + + /* Search for a rectangle which lies "beyond" the current camera + * position + */ + + if (rect->plane > pov->x) + { + /* Get the X distance to the plane. This is an order list: if + * the distance to the plane is larger then the requested step + * then the requested step is acceptable. + */ + + relx = rect->plane - pov->x; + if (relx >= standoff) + { + return reqdeltax; + } + + /* The distance to the plane is smaller that the requested step. + * It may be possible to collide with the plane. Check if a + * collision due to the height of the player is possible first + * (its easier to compute) + */ + + if (footlevel >= rect->vstart && pov->z <= rect->vend) + { + /* A collision is possible based on the players height. + * Now, we'll have to check if a collision is possible + * due to the player's Y position. We can skip this + * step if we are processing another rectangle at the + * same relx distance. + */ + + if (relx != lastrelx) + { + int32_t deltay; /* Scale == "triple" */ + + /* The tangent is equal to the rate of change of Y with + * respect to the X-axis. The tangent is stored at + * double the the "normal" scaling -- so deltay is + * "triple" precision + */ + + deltay = TAN(yaw) * ((int32_t)relx); + absy = tTOs(deltay) + pov->y; /* back to "single" */ + lastrelx = relx; + } + + /* Check if this Y position intersects the rectangle */ + + if (absy >= rect->hstart && absy <= rect->hend) + { + /* It collides -- Check, maybe we can walk through + * this wall + */ + + if (!IS_PASSABLE(rect)) + { + /* Nope... return a clipped value for the new + * player position (which includes the a stand-off + * constant) + * NOTE: This returned value could be negative! + */ + + g_clip_rect = rect; + return relx - MIN_APPROACH_DISTANCE; + } + } + } + } + } + } + + /* The second and third quadrants correspond to the negative X + * direction (excluding angles 90 and 270). + */ + + else if (yaw > ANGLE_90 && yaw < ANGLE_270) + { + /* Calculate the requested distance along the (negative) X axis + * (adding a little to how close the play can come to a wall) + */ + + standoff = MIN_APPROACH_DISTANCE - reqdeltax; + + /* This is the position of the player's feet */ + + footlevel = pov->z - height; + + /* Look at every rectangle lying in the X plane */ + + for (list = g_xplane.tail; list; list = list->blink) + { + rect = &list->d; + + /* Search for a rectangle which lies "before" the current camera + * position + */ + + if (rect->plane < pov->x) + { + /* Get the X distance to the plane. This is an order list: + * if the distance to the plane is larger then the requested + * step then the requested step is acceptable. + */ + + relx = pov->x - rect->plane; + if (relx >= standoff) + { + return reqdeltax; + } + + /* The distance to the plane is smaller that the requested + * step. It may be possible to collide with the plane. Check + * if a collision due to the height of the player is possible + * first (its easier to compute) + */ + + if (footlevel >= rect->vstart && pov->z <= rect->vend) + { + /* A collision is possible based on the players height. + * Now, we'll have to check if a collision is possible due + * to the player's Y position. We can skip this step if + * we are processing another rectangle at the same relx + * distance. + */ + + if (relx != lastrelx) + { + int32_t deltay; /* Scale == "triple" */ + + /* The negative tangent is equal to the rate of change + * of Y with respect to the X-axis.The tangent is + * stored at double the the "normal" scaling -- so + * deltay is "triple" precision + */ + + deltay = -TAN(yaw) * ((int32_t)relx); + absy = tTOs(deltay) + pov->y; /* back to "single" */ + lastrelx = relx; + } + + /* Check if this Y position intersects the rectangle */ + + if (absy >= rect->hstart && absy <= rect->hend) + { + /* It collides -- Check, maybe we can walk through + * this wall? + */ + + if (!IS_PASSABLE(rect)) + { + /* Nope -- return a clipped value for the new + * player position (which includes the a stand-off + * constant) + * NOTE: This returned value could be positive! + */ + + g_clip_rect = rect; + return MIN_APPROACH_DISTANCE - relx; + } + } + } + } + } + } + + /* If we got there, then no collisions were found. Just return the + * requested step value + */ + + return reqdeltax; +} + +/***************************************************************************** + * Name: trv_ray_test_yplane + * + * Description: + * This function tests if there is an Y plane within the supplied distance + * from the player at the supplied angle. This function is used to + * determine in there is a door in front of the player. NOTE: This + * function exploits the normal collision detection logic in + * trv_rayclip_player_ymotion and depends on the side-effect setting of + * g_clip_rect. + * + ****************************************************************************/ + +FAR struct trv_rect_data_s *trv_ray_test_yplane(FAR struct trv_camera_s *pov, + trv_coord_t dist, int16_t yaw, + trv_coord_t height) +{ + (void)trv_rayclip_player_ymotion(pov, dist, yaw, height); + return g_clip_rect; +} + +/***************************************************************************** + * Name: trv_rayclip_player_ymotion + * + * Description: + * This function calculates the acceptable change in the player's position + * along the Y component of the specified yaw angle which would not cause + * a collision with a Y plane. This logic is essentially a modified X + * ray cast. + * + ****************************************************************************/ + +trv_coord_t trv_rayclip_player_ymotion(FAR struct trv_camera_s *pov, + trv_coord_t dist, int16_t yaw, + trv_coord_t height) +{ + FAR struct trv_rect_list_s *list; /* Points to the current Y plane rectangle */ + FAR struct trv_rect_data_s *rect; /* Points to the rectangle data */ + trv_coord_t reqdeltay; + trv_coord_t standoff; + trv_coord_t footlevel; + trv_coord_t rely; /* Relative position of the Y plane */ + trv_coord_t absx; /* Absolute X position at rely given yaw */ + trv_coord_t lastrely = -1; /* Last relative Y position processed */ + + /* Decompose the desired move distance into its Y component */ + + reqdeltay = tTOs(g_sin_table[ yaw ] * dist); + + /* Assume that no clipping will be performed */ + + g_clip_rect = NULL; + + /* Perform Y raycasting based on the quadrant of the yaw angle */ + /* The first and second quadrants correspond to the positive Y + * direction (excluding angles 0 and 180). + */ + + if (yaw > ANGLE_0 && yaw < ANGLE_180) + { + /* Calculate the requested distance along the (positive) X axis + * (adding a little to how close the play can come to a wall) + */ + + standoff = reqdeltay + MIN_APPROACH_DISTANCE; + + /* This is the position of the player's feet */ + + footlevel = pov->z - height; + + /* Look at every rectangle lying in a Y plane */ + + for (list = g_yplane.head; list; list = list->flink) + { + rect = &list->d; + + /* Search for a rectangle which lies "beyond" the current camera + * position + */ + + if (rect->plane > pov->y) + { + /* Get the Y distance to the plane. This is an order list: + * If the distance to the plane is larger then the requested + * step then the requested step is acceptable. + */ + + rely = rect->plane - pov->y; + if (rely >= standoff) + { + return reqdeltay; + } + + /* The distance to the plane is smaller that the requested + * step. It may be possible to collide with the plane. + * Check if a collision due to the height of the player is + * possible first (its easier to compute) + */ + + if (footlevel >= rect->vstart && pov->z <= rect->vend) + { + /* A collision is possible based on the players height. + * Now, we'll have to check if a collision is possible + * due to the player's X position. We can skip this + * step if we are processing another rectangle at the + * same relx distance. + */ + + if (rely != lastrely) + { + int32_t deltax; /* Scale == "triple" */ + + /* The inverted tangent is equal to the rate of change + * of X with respect to the Y-axis. The cotangent is + * stored at double the the "normal" scaling -- so + * deltax is "triple" precision + */ + + deltax = g_cot_table(yaw) * ((int32_t)rely); + absx = tTOs(deltax) + pov->x; /* back to "single" */ + lastrely = rely; + } + + /* Check if this X position intersects the rectangle */ + + if (absx >= rect->hstart && absx <= rect->hend) + { + /* It collides -- Check, maybe we can walk through + * this wall? + */ + + if (!IS_PASSABLE(rect)) + { + /* Nope -- return a clipped value for the new + * player position (which includes the a stand-off + * constant) + * NOTE: This returned value could be negative! + */ + + g_clip_rect = rect; + return rely - MIN_APPROACH_DISTANCE; + } + } + } + } + } + } + + /* The third and fourth quadrants correspond to the negative Y + * direction (excluding angles 0 and 180). + */ + + else if (yaw > ANGLE_180) + { + /* Calculate the requested distance along the (negative) X axis + * (adding a little to how close the play can come to a wall) + */ + + standoff = MIN_APPROACH_DISTANCE - reqdeltay; + + /* This is the position of the player's feet */ + + footlevel = pov->z - height; + + /* Look at every rectangle lying in the X plane */ + + for (list = g_yplane.tail; list; list = list->blink) + { + rect = &list->d; + + /* Search for a rectangle which lies "before" the current camera + * position + */ + + if (rect->plane < pov->y) + { + /* Get the X distance to the plane. This is an order list: if + * the distance to the plane is larger then the requested step + * then the requested step is acceptable. + */ + + rely = pov->y - rect->plane; + if (rely >= standoff) + { + return reqdeltay; + } + + /* The distance to the plane is smaller that the requested + * step. It may be possible to collide with the plane. Check + * if a collision due to the height of the player is possible + * first (its easier to compute) + */ + + if (footlevel >= rect->vstart && pov->z <= rect->vend) + { + /* A collision is possible based on the players height. + * Now, we'll have to check if a collision is possible due + * to the player's X position. We can skip this step if + * we are processing another rectangle at the same relx + * distance. + */ + + if (rely != lastrely) + { + int32_t deltax; /* Scale == "triple" */ + + /* The negative inverted tangent is equal to the rate + * of change of X with respect to the Y-axis. The + * cotangent is stored at double the the "normal" + * scaling -- so deltax is "triple" precision + */ + + deltax = -g_cot_table(yaw - ANGLE_180) * ((int32_t)rely); + absx = tTOs(deltax) + pov->x; /* back to "single" */ + lastrely = rely; + } + + /* Check if this X position intersects the rectangle */ + + if (absx >= rect->hstart && absx <= rect->hend) + { + /* It collides -- Check, maybe we can walk through + * this wall + */ + + if (!IS_PASSABLE(rect)) + { + /* It collides -- return a clipped value for the + * new player position (which includes the a + * stand-off constant) + * NOTE: This returned value could be positive! + */ + + g_clip_rect = rect; + return MIN_APPROACH_DISTANCE - rely; + } + } + } + } + } + } + + /* Return the clipped value */ + + return reqdeltay; +} + +/***************************************************************************** + * Name: trv_ray_adjust_zpos + * + * Description: + * Make sure that the player is standing on something! + * + ****************************************************************************/ + +trv_coord_t trv_ray_adjust_zpos(FAR struct trv_camera_s *pov, + trv_coord_t height) +{ + struct trv_rect_list_s *list; /* Points to the current Z plane rectangle */ + struct trv_rect_data_s *rect; /* Points to the rectangle data */ + + /* We will place the player's feet at the largest Z plane + * which is lower the the player's eye level. We traverse + * the the g_zplane list in order of increase Z values + */ + + for (list = g_zplane.head; list; list = list->flink) + { + rect = &list->d; + + /* The Z plane list is an ordered list. If the next + * plane is over the player's head, then the player + * must be standing at "ground zero." + */ + + if (rect->plane >= pov->z) + { + return height - pov->z; + } + + /* Check if this plane if under the player */ + + if (pov->x >= rect->hstart && pov->x <= rect->hend && + pov->y >= rect->vstart && pov->y <= rect->vend) + { + /* We have the the smallest Z plane under the player + * which is below the player's eye level (pov->z). + * Determine the approach delta Z value to return + */ + + return height + rect->plane - pov->z; + } + } + + /* No plane was found under the player. Set him at his + * height above the "bottom" of the world + */ + + return height - pov->z; +} diff --git a/graphics/traveler/src/trv_raycast.c b/graphics/traveler/src/trv_raycast.c new file mode 100644 index 000000000..88fbe5b72 --- /dev/null +++ b/graphics/traveler/src/trv_raycast.c @@ -0,0 +1,1405 @@ +/******************************************************************************* + * apps/graphics/traveler/src/trv_raycast.c + * This file contains the low-level ray casting logic + * + * Copyright (C) 2014 Gregory Nutt. All rights reserved. + * Author: Gregory Nutt + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in + * the documentation and/or other materials provided with the + * distribution. + * 3. Neither the name NuttX nor the names of its contributors may be + * used to endorse or promote products derived from this software + * without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, + * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, + * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS + * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED + * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN + * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + * + ****************************************************************************/ + +/**************************************************************************** + * Included files + ****************************************************************************/ + +#include "trv_types.h" +#include "trv_world.h" +#include "trv_doors.h" +#include "trv_plane.h" +#include "trv_bitmaps.h" +#include "trv_trigtbl.h" +#include "trv_rayrend.h" +#include "trv_rayprune.h" +#include "trv_raycast.h" + +/**************************************************************************** + * Compilation switches + ****************************************************************************/ + +/**************************************************************************** + * Pre-processor Definitions + ****************************************************************************/ + +/* The following switch enables view correction logic. */ + +#define ENABLE_VIEW_CORRECTION 1 + +/**************************************************************************** + * Private Function Prototypes + ****************************************************************************/ + +static void trv_ray_xcaster14(FAR struct trv_raycast_s *result); +static void trv_ray_xcaster23(FAR struct trv_raycast_s *result); +static void trv_ray_ycaster12(FAR struct trv_raycast_s *result); +static void trv_ray_ycaster34(FAR struct trv_raycast_s *result); +static void trv_ray_zcasteru(FAR struct trv_raycast_s *result); +static void trv_ray_zcasterl(FAR struct trv_raycast_s *result); + +/**************************************************************************** + * Private Data + ****************************************************************************/ + +/* The following are the tangent and the cotangent of the pitch angle + * adjusted for the viewing yaw angle so that the view is correct for the + * "fish eye" effect which results from the projection of the polar ray cast + * onto the flat display + */ + +static int32_t g_adj_tanpitch; +static int32_t g_adj_cotpitch; + +/**************************************************************************** + * Private Functions + ****************************************************************************/ + +/**************************************************************************** + * Name: trv_ray_xcaster14 + * + * Description: + * This function casts a ray along the Y-Axis looking at points of + * intersection on the X-Axis. If a block is defined at this intersection + * then a "hit" is found and the distance to this hit is determined. + * + * If we are in the "right" half of the view, then the X ray cast will + * proceed in a positive along the X axis and that all possible X-axis + * intersections will occur to at the "left" of a cell. + * + * NOTE: The X-Ray caster must run first because it initializes a + * data structure needed by both the Y and Z ray casters. + * + ***************************************************************************/ + +static void trv_ray_xcaster14(FAR struct trv_raycast_s *result) +{ + struct trv_rect_list_s *list; /* Points to the current X plane rectangle */ + struct trv_rect_data_s *rect; /* Points to the rectangle data */ + trv_coord_t relx; /* Relative position of the X plane */ + trv_coord_t absy; /* Absolute Y position at relx given yaw */ + trv_coord_t absz; /* Absolute Z position at relx given pitch */ + trv_coord_t lastrelx1 = -1; /* Last relative X position processed */ + trv_coord_t lastrelx2 = -1; /* Last relative X position processed */ + int32_t dydx; /* Rate of change of Y wrt X (double) */ + int32_t dzdx; /* Rate of change of Z wrt X (double) */ + + /* At a viewing angle of 270 degrees, no intersections with the g_ray_xplanes + * are possible! + */ + + if (g_camera.yaw == ANGLE_270) + { + return; + } + + /* Pre-calculate the rate of change of Y and Z with respect to X */ + /* The tangent is equal to the rate of change of Y with respect to the + * X-axis. The tangent is stored at double the "normal" scaling. + */ + + dydx = TAN(g_camera.yaw); + + /* Determine the rate of change of the Z with respect to X. The tangent is + * "double" precision; the secant is "double" precision. dzdx will be + * retained as "double" precision. + */ + + dzdx = qTOd(g_adj_tanpitch * ABS(g_sec_table[g_camera.yaw])); + + /* Look at every rectangle lying in the X plane */ + /* This logic should be improved at some point so that non-visible planes + * are "pruned" from the list prior to ray casting! + */ + + for (list = g_ray_xplane.head; list; list = list->flink) + { + rect = &list->d; + + /* Search for a rectangle which lies "beyond" the current camera + * position + */ + + if (rect->plane > g_camera.x) + { + /* get the X distance to the plane */ + + relx = rect->plane - g_camera.x; + +#if 0 + /* g_ray_xplane is an ordered list, if we have already hit something + * closer, then we can abort the casting now. + */ + + if (relx > result->xdist) + { + return; + } +#endif + + /* Calculate the Y position at this relative X position. We can skip + * this step if we are processing another rectangle at the same relx + * distance. + */ + + if (relx != lastrelx1) + { + int32_t deltay; /* Scale == "triple" */ + + /* The dydx is stored at double the"normal" scaling -- so + * deltay is "triple" precision + */ + + deltay = dydx * ((int32_t) relx); + absy = tTOs(deltay) + g_camera.y; /* back to "single" */ + lastrelx1 = relx; + } + + /* Check if this Y position intersects the rectangle */ + + if (absy >= rect->hstart && absy <= rect->hend) + { + /* The Y position lies in the rectangle. Now, calculate the + * theZ position at this relative X position. We can skip + * this step if we are processing another rectangle at the + * same relx distance. + */ + + if (relx != lastrelx2) + { + int32_t deltaz; /* Scale == TRIPLE */ + + /* The dzdx is stored at double the"normal" scaling -- so + * deltaz is "triple" precision + */ + + deltaz = dzdx * ((int32_t) relx); + absz = tTOs(deltaz) + g_camera.z; /* Back to single */ + lastrelx2 = relx; + } + + /* Check if this Z position intersects the rectangle */ + + if (absz >= rect->vstart && absz <= rect->vend) + { + /* We've got a potential hit, let's see what it is */ + /* Check if we just hit an ordinary opaque wall */ + + if (IS_NORMAL(rect)) + { + /* Yes..Save the parameters associated with the normal + * wall hit + */ + + result->rect = rect; + result->type = MK_HIT_TYPE(FRONT_HIT, X_HIT); + result->xpos = absy; + result->ypos = absz; + + result->xdist = relx; + result->ydist = ABS(absy - g_camera.y); + result->zdist = ABS(absz - g_camera.z); + + /* Terminate X casting */ + + return; + } + else if (IS_DOOR(rect)) + { + /* Check if the door is in motion. */ + + if (!IS_MOVING_DOOR(rect)) + { + /* Save the parameters associated with the normal + * door hit + */ + + result->rect = rect; + result->type = MK_HIT_TYPE(FRONT_HIT, X_HIT); + result->xpos = absy; + result->ypos = absz; + + result->xdist = relx; + result->ydist = ABS(absy - g_camera.y); + result->zdist = ABS(absz - g_camera.z); + + /* Terminate X casting */ + + return; + } + + /* The door is in motion, the Z-position to see if we can + * see under the door + */ + + else if (absz > g_opendoor.zbottom) + { + /* Save the parameters associated with the moving + * door hit + */ + + result->rect = rect; + result->type = MK_HIT_TYPE(FRONT_HIT, X_HIT); + result->xpos = absy; + result->ypos = absz - g_opendoor.zdist; + + result->xdist = relx; + result->ydist = ABS(absy - g_camera.y); + result->zdist = ABS(absz - g_camera.z); + + /* Terminate X casting */ + + return; + } + } + + /* Otherwise, it must be a transparent wall. We'll need to + * make our decision based upon the pixel that we hit + */ + + /* Check if the pixel at this location is visible */ + + else if (GET_FRONT_PIXEL(rect, absy, absz) != INVISIBLE_PIXEL) + { + /* Its visible, save the parameters associated with the + * transparent wall hit + */ + + result->rect = rect; + result->type = MK_HIT_TYPE(FRONT_HIT, X_HIT); + result->xpos = absy; + result->ypos = absz; + + result->xdist = relx; + result->ydist = ABS(absy - g_camera.y); + result->zdist = ABS(absz - g_camera.z); + + /* Terminate X casting */ + + return; + } + } + } + } + } +} + +/**************************************************************************** + * Name: trv_ray_xcaster23 + * + * Description: + * This function casts a ray along the Y-Axis looking at points of + * intersection on the X-Axis. If a block is defined at this intersection + * then a "hit" is found and the distance to this hit is determined. + * + * If we are in the "left" half of the view, then the X ray cast will + * proceed in a negative along the X axis and that all possible X-axis + * intersections will occur on the "right" of a cell. + * + * NOTE: The X-Ray caster must run first because it initializes a + * data structure needed by both the Y and Z ray casters. + * + ***************************************************************************/ + +static void trv_ray_xcaster23(FAR struct trv_raycast_s *result) +{ + struct trv_rect_list_s *list; /* Points to the current X plane rectangle */ + struct trv_rect_data_s *rect; /* Points to the rectangle data */ + trv_coord_t relx; /* Relative position of the X plane */ + trv_coord_t absy; /* Absolute Y position at relx given yaw */ + trv_coord_t absz; /* Absolute Z position at relx given pitch */ + trv_coord_t lastrelx1 = -1; /* Last relative X position processed */ + trv_coord_t lastrelx2 = -1; /* Last relative X position processed */ + int32_t dydx; /* Rate of change of Y wrt X (double) */ + int32_t dzdx; /* Rate of change of Z wrt X (double) */ + + /* At a view angle of 90 degrees, no intersections with the g_ray_xplanes are + * possible! + */ + + if (g_camera.yaw == ANGLE_90) + { + return; + } + + /* Pre-calculate the rate of change of Y and Z with respect to X */ + /* The negative tangent is equal to the rate of change of Y with respect + * to the X-axis. The tangent is stored at double the "normal" scaling. + */ + + dydx = -TAN(g_camera.yaw); + + /* Determine the rate of change of the Z with respect to X. dydx is + * "double" precision; the secant is "double" precision. dzdx will be + * retained as "double" precision. + */ + + dzdx = qTOd(g_adj_tanpitch * ABS(g_sec_table[g_camera.yaw])); + + /* Look at every rectangle lying in the X plane */ + /* This logic should be improved at some point so that non-visible planes + * are "pruned" from the list prior to ray casting! + */ + + for (list = g_ray_xplane.tail; list; list = list->blink) + { + rect = &list->d; + + /* Search for a rectangle which lies "before" the current camera + * position + */ + + if (rect->plane < g_camera.x) + { + /* get the X distance to the plane */ + + relx = g_camera.x - rect->plane; +#if 0 + /* g_ray_xplane is an ordered list, if we have already hit something + * closer, then we can abort the casting now. + */ + + if (relx > result->xdist) + { + return; + } +#endif + + /* Calculate the Y position at this relative X position. We can skip + * this step if we are processing another rectangle at the same relx + * distance. + */ + + if (relx != lastrelx1) + { + int32_t deltay; /* Scale == "triple" */ + + /* The dydx is stored at double the"normal" scaling -- so deltay + * is "triple" precision + */ + + deltay = dydx * ((int32_t) relx); + absy = tTOs(deltay) + g_camera.y; /* back to "single" */ + lastrelx1 = relx; + } + + /* Check if this Y position intersects the rectangle */ + + if (absy >= rect->hstart && absy <= rect->hend) + { + /* The Y position lies in the rectangle. Now, calculate the + * Z position at this relative X position. We can skip this + * step if we are processing another rectangle at the same + * relx distance. + */ + + if (relx != lastrelx2) + { + int32_t deltaz; /* Scale == TRIPLE */ + + /* The dzdx is stored at double the"normal" scaling -- so + * deltaz is "triple" precision + */ + + deltaz = dzdx * ((int32_t) relx); + absz = tTOs(deltaz) + g_camera.z; /* Back to single */ + lastrelx2 = relx; + } + + /* Check if this Z position intersects the rectangle */ + + if (absz >= rect->vstart && absz <= rect->vend) + { + /* We've got a potential hit, let's see what it is */ + /* Check if we just hit an ordinary opaque wall */ + + if (IS_NORMAL(rect)) + { + /* Yes..Save the parameters associated with the normal + * wall hit + */ + + result->rect = rect; + result->type = MK_HIT_TYPE(BACK_HIT, X_HIT); + result->xpos = absy; + result->ypos = absz; + + result->xdist = relx; + result->ydist = ABS(absy - g_camera.y); + result->zdist = ABS(absz - g_camera.z); + + /* Terminate X casting */ + + return; + } + else if (IS_DOOR(rect)) + { + /* Check if the door is in motion. */ + + if (!IS_MOVING_DOOR(rect)) + { + /* Save the parameters associated with the normal + * door hit + */ + + result->rect = rect; + result->type = MK_HIT_TYPE(BACK_HIT, X_HIT); + result->xpos = absy; + result->ypos = absz; + + result->xdist = relx; + result->ydist = ABS(absy - g_camera.y); + result->zdist = ABS(absz - g_camera.z); + + /* Terminate X casting */ + + return; + } + + /* The door is in motion, the Z-position to see if we can + * see under the door + */ + + else if (absz > g_opendoor.zbottom) + { + /* Save the parameters associated with the moving + * door hit + */ + + result->rect = rect; + result->type = MK_HIT_TYPE(BACK_HIT, X_HIT); + result->xpos = absy; + result->ypos = absz - g_opendoor.zdist; + + result->xdist = relx; + result->ydist = ABS(absy - g_camera.y); + result->zdist = ABS(absz - g_camera.z); + + /* Terminate X casting */ + + return; + } + } + + /* Otherwise, it must be a transparent wall. We'll need to + * make our decision based upon the pixel that we hit + */ + + /* Check if the pixel at this location is visible */ + + else if (GET_BACK_PIXEL(rect, absy, absz) != INVISIBLE_PIXEL) + { + /* Its visible, save the parameters associated with the + * transparent wall hit + */ + + result->rect = rect; + result->type = MK_HIT_TYPE(BACK_HIT, X_HIT); + result->xpos = absy; + result->ypos = absz; + + result->xdist = relx; + result->ydist = ABS(absy - g_camera.y); + result->zdist = ABS(absz - g_camera.z); + + /* Terminate X casting */ + + return; + } + } + } + } + } +} + +/**************************************************************************** + * Name: trv_ray_ycaster12 + * + * Description: + * This function casts a ray along the X-Axis looking at points of + * intersection on the Y-Axis. If a block is defined at this intersection + * then a "hit" is found and the distance to this hit is determined. + * + * If we are in the "forward" half of the view, then the Y ray cast will + * proceed in a positive along the Y axis and that all possible Y-axis + * intersections will occur on the "front" of a cell. + * + * NOTE: The X-Ray is assumed to have been performed first! + * + ***************************************************************************/ + +static void trv_ray_ycaster12(FAR struct trv_raycast_s *result) +{ + struct trv_rect_list_s *list; /* Points to the current P plane rectangle */ + struct trv_rect_data_s *rect; /* Points to the rectangle data */ + trv_coord_t rely; /* Relative position of the Y plane */ + trv_coord_t absx; /* Absolute X position at rely given yaw */ + trv_coord_t absz; /* Absolute Z position at rely given pitch */ + trv_coord_t lastrely1 = -1; /* Last relative Y position processed */ + trv_coord_t lastrely2 = -1; /* Last relative Y position processed */ + int32_t dxdy; /* Rate of change of X wrt Y (double) */ + int32_t dzdy; /* Rate of change of Z wrt Y (double) */ + + /* At a viewing angle of 0 degrees, no intersections with the g_ray_yplane is + * possible! + */ + + if (g_camera.yaw == ANGLE_0) + { + return; + } + + /* Pre-calculate the rate of change of X and Z with respect to Y */ + /* The inverted tangent is equal to the rate of change of X with respect to + * the Y-axis. The cotangent is stored at double the the "normal" scaling. + */ + + dxdy = g_cot_table(g_camera.yaw); + + /* Determine the rate of change of the Z with respect to Y. The tangent + * is "double" precision; the cosecant is "double" precision. dzdy will + * be retained as "double" precision. + */ + + dzdy = qTOd(g_adj_tanpitch * ABS(g_csc_table[g_camera.yaw])); + + /* Look at every rectangle lying in a Y plane */ + /* This logic should be improved at some point so that non-visible planes + * are "pruned" from the list prior to ray casting! + */ + + for (list = g_ray_yplane.head; list; list = list->flink) + { + rect = &list->d; + + /* Search for a rectangle which lies "beyond" the current camera + * position + */ + + if (rect->plane > g_camera.y) + { + /* get the Y distance to the plane */ + + rely = rect->plane - g_camera.y; + + /* g_ray_yplane is an ordered list, if we have already hit something + * closer, then we can abort the casting now. + */ + + if (rely > result->ydist) + { + return; + } + + /* Calculate the Y position at this relative X position. We can skip + * this step if we are processing another rectangle at the same relx + * distance. + */ + + if (rely != lastrely1) + { + int32_t deltax; /* Scale == "triple" */ + + /* The dxdy is stored at double the"normal" scaling -- so deltax + * is "triple" precision + */ + + deltax = dxdy * ((int32_t) rely); + absx = tTOs(deltax) + g_camera.x; /* back to "single" */ + lastrely1 = rely; + } + + /* Check if this X position intersects the rectangle */ + + if (absx >= rect->hstart && absx <= rect->hend) + { + /* The X position lies in the rectangle. Now, calculate the the + * Z position at this relative X position. We can skip this step + * if we are processing another rectangle at the same relx + * distance. + */ + + if (rely != lastrely2) + { + int32_t deltaz; /* Scale == TRIPLE */ + + /* The dzdy is stored at double the"normal" scaling -- so + * deltaz is "triple" precision + */ + + deltaz = dzdy * ((int32_t) rely); + absz = tTOs(deltaz) + g_camera.z; /* Back to single */ + lastrely2 = rely; + } + + /* Check if this Z position intersects the rectangle */ + + if (absz >= rect->vstart && absz <= rect->vend) + { + /* We've got a potential hit, let's see what it is */ + /* Check if we just hit an ordinary opaque wall */ + + if (IS_NORMAL(rect)) + { + /* Yes..Save the parameters associated with the normal + * wall hit + */ + + result->rect = rect; + result->type = MK_HIT_TYPE(FRONT_HIT, Y_HIT); + result->xpos = absx; + result->ypos = absz; + + result->xdist = ABS(absx - g_camera.x); + result->ydist = rely; + result->zdist = ABS(absz - g_camera.z); + + /* Terminate Y casting */ + + return; + } + else if (IS_DOOR(rect)) + { + /* Check if the door is in motion. */ + + if (!IS_MOVING_DOOR(rect)) + { + /* Save the parameters associated with the normal + * door hit + */ + + result->rect = rect; + result->type = MK_HIT_TYPE(FRONT_HIT, Y_HIT); + result->xpos = absx; + result->ypos = absz; + + result->xdist = ABS(absx - g_camera.x); + result->ydist = rely; + result->zdist = ABS(absz - g_camera.z); + + /* Terminate Y casting */ + + return; + } + + /* The door is in motion, the Z-position to see if we can + * see under the door + */ + + else if (absz > g_opendoor.zbottom) + { + /* Save the parameters associated with the moving + * door hit + */ + + result->rect = rect; + result->type = MK_HIT_TYPE(FRONT_HIT, Y_HIT); + result->xpos = absx; + result->ypos = absz - g_opendoor.zdist; + + result->xdist = ABS(absx - g_camera.x); + result->ydist = rely; + result->zdist = ABS(absz - g_camera.z); + + /* Terminate Y casting */ + + return; + } + } + + /* Otherwise, it must be a transparent wall. We'll need to + * make our decision based upon the pixel that we hit + */ + + /* Check if the pixel at this location is visible */ + + else if (GET_FRONT_PIXEL(rect, absx, absz) != INVISIBLE_PIXEL) + { + /* Its visible, save the parameters associated with the + * transparent wall hit + */ + + result->rect = rect; + result->type = MK_HIT_TYPE(FRONT_HIT, Y_HIT); + result->xpos = absx; + result->ypos = absz; + + result->xdist = ABS(absx - g_camera.x); + result->ydist = rely; + result->zdist = ABS(absz - g_camera.z); + + /* Terminate Y casting */ + + return; + } + } + } + } + } +} + +/**************************************************************************** + * Name: trv_ray_ycaster34 + * + * Description: + * This function casts a ray along the X-Axis looking at points of + * intersection on the Y-Axis. If a block is defined at this intersection + * then a "hit" is found and the distance to this hit is determined. + * + * If we are in the "back" half of the view, then the Y ray cast will + * proceed in a negative along the Y axis and that all possible Y-axis + * intersections will occur on the "back" of a cell. + * + * NOTE: The X-Ray is assumed to have been performed first! + * + ***************************************************************************/ + +static void trv_ray_ycaster34(FAR struct trv_raycast_s *result) +{ + struct trv_rect_list_s *list; /* Points to the current P plane rectangle */ + struct trv_rect_data_s *rect; /* Points to the rectangle data */ + trv_coord_t rely; /* Relative position of the Y plane */ + trv_coord_t absx; /* Absolute X position at rely given yaw */ + trv_coord_t absz; /* Absolute Z position at rely given pitch */ + trv_coord_t lastrely1 = -1; /* Last relative Y position processed */ + trv_coord_t lastrely2 = -1; /* Last relative Y position processed */ + int32_t dxdy; /* Rate of change of X wrt Y (double) */ + int32_t dzdy; /* Rate of change of Z wrt Y (double) */ + + /* At a viewing angle of 180 degrees, no intersections with the line x = bXi + * are possible! + */ + + if (g_camera.yaw == ANGLE_180) + { + return; + } + + /* Pre-calculate the rate of change of X and Z with respect to Y */ + /* The negative inverted tangent is equal to the rate of change of X with + * respect to the Y-axis. The cotangent is stored at double the the + * "normal" scaling. + */ + + dxdy = -g_cot_table(g_camera.yaw - ANGLE_180); + + /* Determine the rate of change of the Z with respect to Y. The tangent + * is "double" precision; the cosecant is "double" precision. dzdy will + * be retained as "double" precision. + */ + + dzdy = qTOd(g_adj_tanpitch * ABS(g_csc_table[g_camera.yaw])); + + /* Look at every rectangle lying in a Y plane */ + /* This logic should be improved at some point so that non-visible planes + * are "pruned" from the list prior to ray casting! + */ + + for (list = g_ray_yplane.tail; list; list = list->blink) + { + rect = &list->d; + + /* Search for a rectangle which lies "before" the current camera + * position + */ + + if (rect->plane < g_camera.y) + { + /* get the Y distance to the plane */ + + rely = g_camera.y - rect->plane; + + /* g_ray_yplane is an ordered list, if we have already hit something + * closer, then we can abort the casting now. + */ + + if (rely > result->ydist) + { + return; + } + + /* Calculate the Y position at this relative X position. We can skip + * this step if we are processing another rectangle at the same relx + * distance. + */ + + if (rely != lastrely1) + { + int32_t deltax; /* Scale == "triple" */ + + /* The dxdy is stored at double the"normal" scaling -- so deltax + * is "triple" precision + */ + + deltax = dxdy * ((int32_t) rely); + absx = tTOs(deltax) + g_camera.x; /* back to "single" */ + lastrely1 = rely; + } + + /* Check if this X position intersects the rectangle */ + + if (absx >= rect->hstart && absx <= rect->hend) + { + + /* The X position lies in the rectangle. Now, calculate the + * Z position at this relative X position. We can skip this + * step if we are processing another rectangle at the same + * relx distance. + */ + + if (rely != lastrely2) + { + int32_t deltaz; /* Scale == TRIPLE */ + + /* The dzdy is stored at double the"normal" scaling -- so + * deltaz is "triple" precision + */ + + deltaz = dzdy * ((int32_t) rely); + absz = tTOs(deltaz) + g_camera.z; /* Back to single */ + lastrely2 = rely; + } + + /* Check if this Z position intersects the rectangle */ + + if (absz >= rect->vstart && absz <= rect->vend) + { + + /* We've got a potential hit, let's see what it is */ + /* Check if we just hit an ordinary opaque wall */ + + if (IS_NORMAL(rect)) + { + /* Yes..Save the parameters associated with the normal + * wall hit + */ + + result->rect = rect; + result->type = MK_HIT_TYPE(BACK_HIT, Y_HIT); + result->xpos = absx; + result->ypos = absz; + + result->xdist = ABS(absx - g_camera.x); + result->ydist = rely; + result->zdist = ABS(absz - g_camera.z); + + /* Terminate Y casting */ + + return; + } + else if (IS_DOOR(rect)) + { + /* Check if the door is in motion. */ + + if (!IS_MOVING_DOOR(rect)) + { + /* Save the parameters associated with the normal + * door hit + */ + + result->rect = rect; + result->type = MK_HIT_TYPE(BACK_HIT, Y_HIT); + result->xpos = absx; + result->ypos = absz; + + result->xdist = ABS(absx - g_camera.x); + result->ydist = rely; + result->zdist = ABS(absz - g_camera.z); + + /* Terminate Y casting */ + + return; + } + + /* The door is in motion, the Z-position to see if we can + * see under the door + */ + + else if (absz > g_opendoor.zbottom) + { + /* Save the parameters associated with the moving + * door hit + */ + + result->rect = rect; + result->type = MK_HIT_TYPE(BACK_HIT, Y_HIT); + result->xpos = absx; + result->ypos = absz - g_opendoor.zdist; + + result->xdist = ABS(absx - g_camera.x); + result->ydist = rely; + result->zdist = ABS(absz - g_camera.z); + + /* Terminate Y casting */ + + return; + } + } + + /* Otherwise, it must be a transparent wall. We'll need to + * make our decision based upon the pixel that we hit + */ + + /* Check if the pixel at this location is visible */ + + else if (GET_BACK_PIXEL(rect, absx, absz) != INVISIBLE_PIXEL) + { + /* Its visible, save the parameters associated with the + * transparent wall hit + */ + + result->rect = rect; + result->type = MK_HIT_TYPE(BACK_HIT, Y_HIT); + result->xpos = absx; + result->ypos = absz; + + result->xdist = ABS(absx - g_camera.x); + result->ydist = rely; + result->zdist = ABS(absz - g_camera.z); + + /* Terminate Y casting */ + + return; + } + } + } + } + } +} + +/**************************************************************************** + * Name: trv_ray_zcasteru + * + * Description: + * This function casts a ray along the rotated Y-Axis looking at points of + * intersection on the Z-Axis. If a block is defined at this intersection + * then a "hit" is found and the distance to this hit is determined. + * + * If we are in the "upper" half of the view, then the Z ray cast will + * proceed along the positive Z axis and that all possible Z-axis + * intersections will occur on the "bottom" of a cell. + * + * NOTE: It is assumed that both the X and Y ray casters have already + * ran! + ***************************************************************************/ + +static void trv_ray_zcasteru(FAR struct trv_raycast_s *result) +{ + struct trv_rect_list_s *list; /* Points to the current Z plane rectangle */ + struct trv_rect_data_s *rect; /* Points to the rectangle data */ + trv_coord_t relz; /* Relative position of the Z plane */ + trv_coord_t absx; /* Absolute X position at relz given yaw */ + trv_coord_t absy; /* Absolute Y position at relz given yaw */ + trv_coord_t lastrelz1 = -1; /* Last relative Z position processed */ + trv_coord_t lastrelz2 = -1; /* Last relative Z position processed */ + int32_t dxdz; /* Rate of change of X wrt Z (double) */ + int32_t dydz; /* Rate of change of Y wrt Z (double) */ + + /* At a viewing angle of 0 degrees, no intersections with the g_ray_zplanes are + * possible! + */ + + if (g_camera.pitch == ANGLE_0) + { + return; + } + + /* Pre-calculate the rate of change of X and Y with respect to Z */ + /* Calculate the rate of change of X with respect to the Z-axis. The + * cotangent is stored at double the "normal" scaling and the cosine is + * also at double scaling. dxdz will be also be stored at double + * precision. + */ + + dxdz = qTOd(g_adj_cotpitch * ((int32_t) g_cos_table[g_camera.yaw])); + + /* Calculate the rate of change of Y with respect to the Z-axis. The + * cotangent is stored at double the "normal" scaling and the sine is also + * at double scaling. dxdz will be also be stored at double precision. + */ + + dydz = qTOd(g_adj_cotpitch * ((int32_t) g_sin_table[g_camera.yaw])); + + /* Look at every rectangle lying in the Z plane */ + /* This logic should be improved at some point so that non-visible planes + * are "pruned" from the list prior to ray casting! + */ + + for (list = g_ray_zplane.head; list; list = list->flink) + { + rect = &list->d; + + /* Search for a rectangle which lies "beyond" the current camera + * position + */ + + if (rect->plane > g_camera.z) + { + /* get the Z distance to the plane */ + + relz = rect->plane - g_camera.z; + + /* g_ray_zplane is an ordered list, if we have already hit something + * closer, then we can abort the casting now. + */ + + if (relz > result->zdist) + { + return; + } + + /* Calculate the X position at this relative Z position. We can skip + * this step if we are processing another rectangle at the same relx + * distance. + */ + + if (relz != lastrelz1) + { + int32_t deltax; /* Scale == "triple" */ + + /* The dxdz is stored at double the"normal" scaling -- so deltax + * is "triple" precision + */ + + deltax = dxdz * ((int32_t) relz); + absx = tTOs(deltax) + g_camera.x; /* back to "single" */ + lastrelz1 = relz; + } + + /* Check if this X position intersects the rectangle */ + + if (absx >= rect->hstart && absx <= rect->hend) + { + /* The X position lies in the rectangle. Now, calculate the the + * Y position at this relative Z position. We can skip this step + * if we are processing another rectangle at the same relx + * distance. + */ + + if (relz != lastrelz2) + { + int32_t deltay; /* Scale == TRIPLE */ + + /* The dydz is stored at double the"normal" scaling -- so + * deltay is "triple" precision + */ + + deltay = dydz * ((int32_t) relz); + absy = tTOs(deltay) + g_camera.y; /* back to "single" */ + lastrelz2 = relz; + } + + /* Check if this Y position intersects the rectangle */ + + if (absy >= rect->vstart && absy <= rect->vend) + { + /* We've got a hit, ..Save the parameters associated with the + * ceiling hit + */ + + result->rect = rect; + result->type = MK_HIT_TYPE(BACK_HIT, Z_HIT); + result->xpos = absx; + result->ypos = absy; + + result->xdist = ABS(absx - g_camera.x); + result->ydist = ABS(absy - g_camera.y); + result->zdist = relz; + + /* Terminate Z casting */ + + return; + } + } + } + } +} + +/**************************************************************************** + * Name: trv_ray_zcasterl + * + * Description: + * This function casts a ray along the rotated Y-Axis looking at points of + * intersection on the Z-Axis. If a block is defined at this intersection + * then a "hit" is found and the distance to this hit is determined. + * + * If we are in the "lower" half of the view, then the Z ray cast will + * proceed along the negative Z axis and that all possible Z-axis + * intersections will occur on the "top" of a cell. + * + * NOTE: It is assumed that both the X and Y ray casters have already + * ran! + * + ***************************************************************************/ + +static void trv_ray_zcasterl(FAR struct trv_raycast_s *result) +{ + struct trv_rect_list_s *list; /* Points to the current Z plane rectangle */ + struct trv_rect_data_s *rect; /* Points to the rectangle data */ + trv_coord_t relz; /* Relative position of the Z plane */ + trv_coord_t absx; /* Absolute X position at relz given yaw */ + trv_coord_t absy; /* Absolute Y position at relz given yaw */ + trv_coord_t lastrelz1 = -1; /* Last relative Z position processed */ + trv_coord_t lastrelz2 = -1; /* Last relative Z position processed */ + int32_t dxdz; /* Rate of change of X wrt Z (double) */ + int32_t dydz; /* Rate of change of Y wrt Z (double) */ + + /* At a viewing angle of 0 degrees, no intersections with the g_ray_zplanes are + * possible! + */ + + if (g_camera.pitch == ANGLE_0) + { + return; + } + + /* Pre-calculate the rate of change of X and Y with respect to Z */ + /* Calculate the rate of change of X with respect to the Z-axis. The + * cotangent is stored at double the "normal" scaling and the cosine is + * also at double scaling. dxdz will be also be stored at double + * precision. + */ + + dxdz = qTOd(g_adj_cotpitch * ((int32_t) g_cos_table[g_camera.yaw])); + + /* Calculate the rate of change of Y with respect to the Z-axis. The + * cotangent is stored at double the "normal" scaling and the sine is + * also at double scaling. dxdz will be also be stored at double + * precision. + */ + + dydz = qTOd(g_adj_cotpitch * ((int32_t) g_sin_table[g_camera.yaw])); + + /* Look at every rectangle lying in the Z plane */ + /* This logic should be improved at some point so that non-visible planes + * are "pruned" from the list prior to ray casting! + */ + + for (list = g_ray_zplane.tail; list; list = list->blink) + { + rect = &list->d; + + /* Search for a rectangle which lies "before" the current camera + * position + */ + + if (rect->plane < g_camera.z) + { + /* get the Z distance to the plane */ + + relz = g_camera.z - rect->plane; + + /* g_ray_zplane is an ordered list, if we have already hit something + * closer, then we can abort the casting now. + */ + + if (relz > result->zdist) + { + return; + } + + /* Calculate the X position at this relative Z position. We can skip + * this step if we are processing another rectangle at the same relx + * distance. + */ + + if (relz != lastrelz1) + { + int32_t deltax; /* Scale == "triple" */ + + /* The dxdz is stored at double the"normal" scaling -- so deltax + * is "triple" precision + */ + + deltax = dxdz * ((int32_t) relz); + absx = tTOs(deltax) + g_camera.x; /* back to "single" */ + lastrelz1 = relz; + } + + /* Check if this X position intersects the rectangle */ + + if (absx >= rect->hstart && absx <= rect->hend) + { + /* The X position lies in the rectangle. Now, calculate the the + * Y position at this relative Z position. We can skip this step + * if we are processing another rectangle at the same relx + * distance. + */ + + if (relz != lastrelz2) + { + int32_t deltay; /* Scale == TRIPLE */ + + /* The dydz is stored at double the"normal" scaling -- so + * deltay is "triple" precision + */ + + deltay = dydz * ((int32_t) relz); + absy = tTOs(deltay) + g_camera.y; /* back to "single" */ + lastrelz2 = relz; + } + + /* Check if this Y position intersects the rectangle */ + + if (absy >= rect->vstart && absy <= rect->vend) + { + /* We've got a hit, ..Save the parameters associated with the + * floor hit + */ + + result->rect = rect; + result->type = MK_HIT_TYPE(FRONT_HIT, Z_HIT); + result->xpos = absx; + result->ypos = absy; + + result->xdist = ABS(absx - g_camera.x); + result->ydist = ABS(absy - g_camera.y); + result->zdist = relz; + + /* Terminate Z casting */ + + return; + } + } + } + } +} + +/**************************************************************************** + * Public Functions + ****************************************************************************/ + +/**************************************************************************** + * Name: trv_raycast + * + * Description: + * This function performs a single ray cast. It decomposes the cast by + * quadrants so that simpler casting algorithms can be used. It also + * enforces the order of casting: X first, then Y, and finally Z. + * + ***************************************************************************/ + +void trv_raycast(int16_t pitch, int16_t yaw, int16_t screenyaw, + FAR struct trv_raycast_s *result) +{ + /* Set the camera pitch and yaw angles for this cast */ + + g_camera.pitch = pitch; + g_camera.yaw = yaw; + + /* Initialize the result structure, assuming that there will be no hit */ + + result->rect = NULL; + result->type = NO_HIT; + result->xpos = 0; + result->ypos = 0; + result->xdist = TRV_INFINITY; + result->ydist = TRV_INFINITY; + result->zdist = TRV_INFINITY; + + /* Prepare for X and Y ray casts. These casts will need the adjusted tangent + * of the pitch angle in order to correct for "fish eye" distortion. This + * correction consists of multiplying by the cosine of the relative screen + * yaw position. The tangent is double precision, the cosine is double + * precision, the result will be retained as double precision. + */ + + screenyaw = ABS(screenyaw); +#if ENABLE_VIEW_CORRECTION + g_adj_tanpitch = qTOd(TAN(pitch) * ((int32_t) g_cos_table[screenyaw])); +#else + g_adj_tanpitch = TAN(pitch); +#endif + + /* Perform X & Y raycasting based on the quadrant of the yaw angle */ + + if (g_camera.yaw < ANGLE_90) + { + trv_ray_xcaster14(result); + trv_ray_ycaster12(result); + } + else if (g_camera.yaw < ANGLE_180) + { + trv_ray_xcaster23(result); + trv_ray_ycaster12(result); + } + else if (g_camera.yaw < ANGLE_270) + { + trv_ray_xcaster23(result); + trv_ray_ycaster34(result); + } + else + { + trv_ray_xcaster14(result); + trv_ray_ycaster34(result); + } + + /* Perform Z ray casting based upon if we are looking up or down */ + + if (g_camera.pitch < ANGLE_90) + { + /* Get the adjusted cotangent of the pitch angle which is used to correct + * for the "fish eye" distortion. This correction consists of + * multiplying by the inverted cosine of the relative screen yaw + * position. The cotangent is double precision, the secant is double + * precision, the result will be retained as double precision. + */ + +#if ENABLE_VIEW_CORRECTION + g_adj_cotpitch = qTOd(g_cot_table(pitch) * g_sec_table[screenyaw]); +#else + g_adj_cotpitch = g_cot_table(pitch); +#endif + trv_ray_zcasteru(result); + } + else + { + /* Get the adjusted cotangent of the pitch angle which is used to correct + * for the "fish eye" distortion. This correction consists of + * multiplying by the inverted cosine of the relative screen yaw + * position. The cotangent is double precision, the secant is double + * precision, the result will be retained as double precision. + */ + +#if ENABLE_VIEW_CORRECTION + g_adj_cotpitch = + qTOd(g_cot_table(ANGLE_360 - pitch) * g_sec_table[screenyaw]); +#else + g_adj_cotpitch = g_cot_table(ANGLE_360 - pitch); +#endif + trv_ray_zcasterl(result); + } +} diff --git a/graphics/traveler/src/trv_raycntl.c b/graphics/traveler/src/trv_raycntl.c new file mode 100644 index 000000000..f1f8b1cec --- /dev/null +++ b/graphics/traveler/src/trv_raycntl.c @@ -0,0 +1,805 @@ +/******************************************************************************* + * apps/graphics/traveler/src/trv_raycntl.c + * This file contains the high-level ray caster control logic + * + * Copyright (C) 2014 Gregory Nutt. All rights reserved. + * Author: Gregory Nutt + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in + * the documentation and/or other materials provided with the + * distribution. + * 3. Neither the name NuttX nor the names of its contributors may be + * used to endorse or promote products derived from this software + * without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, + * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, + * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS + * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED + * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN + * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + * + ****************************************************************************/ + +/**************************************************************************** + * Included files + ****************************************************************************/ + +#include "trv_types.h" +#include "trv_debug.h" +#include "trv_world.h" +#include "trv_plane.h" +#include "trv_bitmaps.h" +#include "trv_paltable.h" +#include "trv_trigtbl.h" +#include "trv_graphics.h" +#include "trv_pov.h" +#include "trv_rayrend.h" +#include "trv_rayprune.h" +#include "trv_raycast.h" +#include "trv_raycntl.h" + +/**************************************************************************** + * Pre-processor Definitions + ****************************************************************************/ + +/* These definitions simplify creation of the initial ray casting cell */ + +#define TOP_HEIGHT (VGULP_SIZE/2) +#define BOT_HEIGHT (VGULP_SIZE - TOP_HEIGHT + 1) + +#define TOP_ROW 0 +#define MID_ROW (TOP_ROW + TOP_HEIGHT - 1) +#define BOT_ROW (VGULP_SIZE-1) + +#define LEFT_WIDTH ((HGULP_SIZE+1)/2) +#define RIGHT_WIDTH (HGULP_SIZE - LEFT_WIDTH + 2) + +#define LEFT_COL 0 +#define MID_COL (LEFT_COL + LEFT_WIDTH - 1) +#define RIGHT_COL HGULP_SIZE + +/* The following macro converts a g_yaw[] array index into a relative screen + * yaw. This is used by the ray caster to perform view corrections. + */ + +#define RELYAW(i) (((i) + WINDOW_LEFT) - (WINDOW_WIDTH/2)) + +/* Macro to determine if two hits "hit" the same object */ + +#define SAME_CELL(i1,j1,i2,j2) \ + (g_ray_hit[i1][j1].rect == g_ray_hit[i2][j2].rect) + +/**************************************************************************** + * Public Data + ****************************************************************************/ + +/* The following array describes the hits from X/Y/Z-ray casting for the + * current HGULP_SIZE x VGULP_SIZE cell + */ + +struct trv_raycast_s g_ray_hit[VGULP_SIZE][HGULP_SIZE + 1]; + +/* This array points to the screen buffer row corresponding to the + * pitch angle + */ + +FAR uint8_t *g_buffer_row[VGULP_SIZE]; + +/* The is the "column" offset in g_buffer_row for the current cell being + * operated on. This value is updated in a loop by trv_raycaster. */ + +int16_t g_cell_column; + +/* This structure holds the parameters used in the current ray cast */ + +struct trv_camera_s g_camera; + +/**************************************************************************** + * Private Data + ****************************************************************************/ + +/* These are all of the yaw angles which will be used by the ray caster + * on a given cycle + */ + +static int16_t g_yaw[IMAGE_WIDTH + 1]; + +/* These are all of the pitch angles which will be used by the ray caster + * on each horizontal pass of a given cycle + */ + +static int16_t g_pitch[VGULP_SIZE]; + +/**************************************************************************** + * Private Functions + ****************************************************************************/ + +/**************************************************************************** + * Function: trv_resolve_cell + * + * Description: + * If the hits in the corners of the input cell are not the same, then + * recursively resolve the cell until all four corners are the same. + * When the are the same, then rend the cell to the display buffer. + * + ***************************************************************************/ + +static void trv_resolve_cell(uint8_t toprow, uint8_t leftcol, + uint8_t height, uint8_t width) +{ + uint8_t midrow; + uint8_t botrow; + uint8_t midcol; + uint8_t rightcol; + uint8_t topheight; + uint8_t botheight; + uint8_t leftwidth; + uint8_t rightwidth; + + /* Check if the cell has been reduced to a vertical line */ + + if (width > 1) + { + /* No.. Check if the cell has been reduce to a horizontal line */ + + if (height > 1) + { + /* No.. It is still a rectangular region. Check if the top half hit + * the same cell type + */ + + if (!SAME_CELL(toprow, leftcol, toprow, (leftcol + width - 1))) + { + /* No.. the top corners are different. Compare the top left and + * bottom left corners to decide how to divide this up + */ + + if (!SAME_CELL(toprow, leftcol, (toprow + height - 1), leftcol)) + { + /* The left corners are not the same. Check the right + * corners. + */ + + if (!SAME_CELL(toprow, (leftcol + width - 1), + (toprow + height - 1), (leftcol + width - 1))) + { + /* The right corners are not the same either. Divide the + * cell into three cells, retaining the bottom half + * (whose corners might be the same). + */ + + leftwidth = ((width + 1) >> 1); + if (leftwidth == 1) + { + rightwidth = 1; + midcol = leftcol + 1; + } + else + { + /* The cell is greater than 2 columns in width */ + + rightwidth = width - leftwidth + 1; + midcol = leftcol + leftwidth - 1; + + /* Get the top middle hit */ + + trv_raycast(g_pitch[toprow], + g_yaw[g_cell_column + midcol], + RELYAW(g_cell_column + midcol), + &g_ray_hit[toprow][midcol]); + } + + topheight = ((height + 1) >> 1); + if (topheight == 1) + { + botheight = 1; + midrow = toprow + 1; + } + else + { + /* The cell is greater that 2 rows in height */ + + botheight = height - topheight + 1; + midrow = toprow + topheight - 1; + + /* Get the middle left hit */ + + trv_raycast(g_pitch[midrow], + g_yaw[g_cell_column + leftcol], + RELYAW(g_cell_column + leftcol), + &g_ray_hit[midrow][leftcol]); + + /* Get the center hit */ + + if (rightwidth > 1) + { + trv_raycast(g_pitch[midrow], + g_yaw[g_cell_column + midcol], + RELYAW(g_cell_column + midcol), + &g_ray_hit[midrow][midcol]); + } + + /* Get the middle right hit */ + + rightcol = leftcol + width - 1; + trv_raycast(g_pitch[midrow], + g_yaw[g_cell_column + rightcol], + RELYAW(g_cell_column + rightcol), + &g_ray_hit[midrow][rightcol]); + } + + trv_resolve_cell(toprow, leftcol, topheight, leftwidth); + trv_resolve_cell(toprow, midcol, topheight, rightwidth); + trv_resolve_cell(midrow, leftcol, botheight, width); + } + + /* The left corners are not the same, but the right are. + * Divide the cell into three cells, retaining the right half + */ + + else + { + leftwidth = ((width + 1) >> 1); + if (leftwidth == 1) + { + rightwidth = 1; + midcol = leftcol + 1; + } + else + { + /* The cell is greater than 2 columns in width */ + + rightwidth = width - leftwidth + 1; + midcol = leftcol + leftwidth - 1; + + /* Get the top middle hit */ + + trv_raycast(g_pitch[toprow], + g_yaw[g_cell_column + midcol], + RELYAW(g_cell_column + midcol), + &g_ray_hit[toprow][midcol]); + + /* Get the bottom middle hit */ + + botrow = toprow + height - 1; + trv_raycast(g_pitch[botrow], + g_yaw[g_cell_column + midcol], + RELYAW(g_cell_column + midcol), + &g_ray_hit[botrow][midcol]); + } + + topheight = ((height + 1) >> 1); + if (topheight == 1) + { + botheight = 1; + midrow = toprow + 1; + } + else + { + /* The cell is greater that 2 rows in height */ + + botheight = height - topheight + 1; + midrow = toprow + topheight - 1; + + /* Get the middle left hit */ + + trv_raycast(g_pitch[midrow], + g_yaw[g_cell_column + leftcol], + RELYAW(g_cell_column + leftcol), + &g_ray_hit[midrow][leftcol]); + + /* Get the center hit */ + + if (rightwidth > 1) + { + trv_raycast(g_pitch[midrow], + g_yaw[g_cell_column + midcol], + RELYAW(g_cell_column + midcol), + &g_ray_hit[midrow][midcol]); + } + } + + trv_resolve_cell(toprow, leftcol, topheight, leftwidth); + trv_resolve_cell(midrow, leftcol, botheight, leftwidth); + trv_resolve_cell(toprow, midcol, height, rightwidth); + } + } + + /* The left sides are the same! Divide the cell vertically into + * two cells */ + + else + { + leftwidth = ((width + 1) >> 1); + if (leftwidth == 1) + { + rightwidth = 1; + midcol = leftcol + 1; + } + else + { + /* The cell is greater than 2 columns in width */ + + rightwidth = width - leftwidth + 1; + midcol = leftcol + leftwidth - 1; + + /* Get the top middle hit */ + + trv_raycast(g_pitch[toprow], g_yaw[g_cell_column + midcol], + RELYAW(g_cell_column + midcol), + &g_ray_hit[toprow][midcol]); + + /* Get the bottom middle hit */ + + botrow = toprow + height - 1; + trv_raycast(g_pitch[botrow], g_yaw[g_cell_column + midcol], + RELYAW(g_cell_column + midcol), + &g_ray_hit[botrow][midcol]); + } + + trv_resolve_cell(toprow, leftcol, height, leftwidth); + trv_resolve_cell(toprow, midcol, height, rightwidth); + } + } + + /* The top corners are the same. Compare the top left and bottom + * left corners + */ + + else if (!SAME_CELL(toprow, leftcol, (toprow + height - 1), leftcol)) + { + /* The top corners are the same, but left corners are not. Divide + * the cell into two cells horizontally + */ + + topheight = ((height + 1) >> 1); + if (topheight == 1) + { + botheight = 1; + midrow = toprow + 1; + } + else + { + /* The cell is greater that 2 rows in height */ + + botheight = height - topheight + 1; + midrow = toprow + topheight - 1; + + /* Get the middle left hit */ + + trv_raycast(g_pitch[midrow], g_yaw[g_cell_column + leftcol], + RELYAW(g_cell_column + leftcol), + &g_ray_hit[midrow][leftcol]); + + /* Get the middle right hit */ + + rightcol = leftcol + width - 1; + trv_raycast(g_pitch[midrow], g_yaw[g_cell_column + rightcol], + RELYAW(g_cell_column + rightcol), + &g_ray_hit[midrow][rightcol]); + } + + trv_resolve_cell(toprow, leftcol, topheight, width); + trv_resolve_cell(midrow, leftcol, botheight, width); + } + + /* The top and left corners are the same. Check the lower right + * corner + */ + + else if (!SAME_CELL(toprow, leftcol, (toprow + height - 1), (leftcol + width - 1))) + { + /* The lower right corner differs from all of the others. Divide + * the cell into three cells, retaining the left half + */ + + leftwidth = ((width + 1) >> 1); + if (leftwidth == 1) + { + rightwidth = 1; + midcol = leftcol + 1; + } + else + { + /* The cell is greater than 2 columns in width */ + + rightwidth = width - leftwidth + 1; + midcol = leftcol + leftwidth - 1; + + /* Get the top middle hit */ + + trv_raycast(g_pitch[toprow], g_yaw[g_cell_column + midcol], + RELYAW(g_cell_column + midcol), + &g_ray_hit[toprow][midcol]); + + /* Get the bottom middle hit */ + + botrow = toprow + height - 1; + trv_raycast(g_pitch[botrow], g_yaw[g_cell_column + midcol], + RELYAW(g_cell_column + midcol), + &g_ray_hit[botrow][midcol]); + } + + topheight = ((height + 1) >> 1); + if (topheight == 1) + { + botheight = 1; + midrow = toprow + 1; + } + else + { + /* The cell is greater that 2 rows in height */ + + botheight = height - topheight + 1; + midrow = toprow + topheight - 1; + + /* Get the middle right hit */ + + rightcol = leftcol + width - 1; + trv_raycast(g_pitch[midrow], g_yaw[g_cell_column + rightcol], + RELYAW(g_cell_column + rightcol), + &g_ray_hit[midrow][rightcol]); + + /* Get the center hit */ + + if (rightwidth > 1) + { + trv_raycast(g_pitch[midrow], g_yaw[g_cell_column + midcol], + RELYAW(g_cell_column + midcol), + &g_ray_hit[midrow][midcol]); + } + } + + trv_resolve_cell(toprow, leftcol, height, leftwidth); + trv_resolve_cell(toprow, midcol, topheight, rightwidth); + trv_resolve_cell(midrow, midcol, botheight, rightwidth); + } + + /* The four corners are the same! */ + + else + { + /* Apply texturing */ + + trv_rend_cell(toprow, leftcol, height, width); + } + } + + /* The cell has been reduced to a horizontal line */ + + else + { + /* Check if the endpoints of the horizontal line are the same */ + + if (!SAME_CELL(toprow, leftcol, toprow, (leftcol + width - 1))) + { + /* No.. they are different. Divide the line in half */ + + leftwidth = ((width + 1) >> 1); + if (leftwidth == 1) + { + rightwidth = 1; + midcol = leftcol + 1; + } + else + { + /* The cell is greater than 2 columns in width */ + + rightwidth = width - leftwidth + 1; + midcol = leftcol + leftwidth - 1; + + /* Get the middle hit */ + + trv_raycast(g_pitch[toprow], g_yaw[g_cell_column + midcol], + RELYAW(g_cell_column + midcol), + &g_ray_hit[toprow][midcol]); + } + + trv_resolve_cell(toprow, leftcol, 1, leftwidth); + trv_resolve_cell(toprow, midcol, 1, rightwidth); + } + + /* The endpoints of the horizontal line are the same! */ + + else + { + /* Apply texturing */ + + trv_rend_row(toprow, leftcol, width); + } + } + } + + /* The cell is only 1 pixel wide. Check if it has been reduced to a single + * pixel + */ + + else if (height > 1) + { + /* No.. The cell has been reduced to a vertical line. Check if the + * endpoints are the same. + */ + + if (!SAME_CELL(toprow, leftcol, (toprow + height - 1), leftcol)) + { + /* No.. they are different. Divide the line in half */ + + topheight = ((height + 1) >> 1); + if (topheight == 1) + { + botheight = 1; + midrow = toprow + 1; + } + else + { + /* The cell is greater that 2 rows in height */ + + botheight = height - topheight + 1; + midrow = toprow + topheight - 1; + + /* Get the middle hit */ + + trv_raycast(g_pitch[midrow], g_yaw[g_cell_column + leftcol], + RELYAW(g_cell_column + leftcol), + &g_ray_hit[midrow][leftcol]); + } + + trv_resolve_cell(toprow, leftcol, topheight, 1); + trv_resolve_cell(midrow, leftcol, botheight, 1); + } + + /* The endpoints of the vertical line are the same! */ + + else + { + /* Apply texturing */ + + trv_rend_column(toprow, leftcol, height); + } + } + + /* The cell has been reduced to a single pixel. */ + + else + { + /* Apply texturing */ + + trv_rend_pixel(toprow, leftcol); + } +} + +/**************************************************************************** + * Public Functions + ****************************************************************************/ + +/**************************************************************************** + * Function: trv_raycaster + * + * Description: + * This is the heart of the system. it casts out 320 rays and builds the + * 3-D image from their intersections with the walls. + * + ***************************************************************************/ + +void trv_raycaster(FAR struct trv_camera_s *player, + FAR struct trv_graphics_info_s *ginfo) +{ + int16_t row; /* the current row being cast 0..IMAGE_HEIGHT */ + int16_t yaw; /* Working yaw angle */ + int16_t pitch; /* Working pitch angle */ + uint8_t *buffer; /* Points the screen buffer row for this pitch */ + int i; + + trv_vdebug("\ntrv_raycaster: x=%d y=%d z=%d yaw=%d pitch=%d", + player->x, player->x, player->z, player->yaw, player->pitch); + + /* Copy the input "player" to the working "camera */ + + g_camera = *player; + + /* The horizontal field of view is determined by the width of the window + * (centered at the yaw angle) + */ + + yaw = g_camera.yaw - (IMAGE_WIDTH / 2) + (HGULP_SIZE * VIDEO_COLUMN_ANGLE); + if (yaw < 0) + { + yaw += ANGLE_360; + } + + /* The vertical field of view is determined by the currently active screen + * HEIGHT. NOTE: camera pitch cannot exceed 90 degrees. Not check is + * performed here. + */ + + pitch = g_camera.pitch + (IMAGE_HEIGHT / 2); + if (pitch >= ANGLE_360) + { + pitch -= ANGLE_360; + } + + /* Loop through all columns at each yaw angle on the screen */ + + for (g_cell_column = IMAGE_WIDTH; g_cell_column >= 0; g_cell_column--) + { + /* Save the yaw angle. By saving all of the yaw angles, we can avoid + * complex tests for 360 degree wraps. + */ + + g_yaw[g_cell_column] = yaw; + + /* Test if viewing yaw angle needs to wrap around */ + + yaw += VIDEO_COLUMN_ANGLE; + if (yaw >= ANGLE_360) + { + yaw -= ANGLE_360; + } + } + + /* Seed the algorithm PART I: Set up the raycaster this yaw range. */ + + trv_ray_yawprune(g_yaw[IMAGE_WIDTH], g_yaw[0]); + + /* Top of Ray Casting Loops */ + + /* Point to the first row of the rending buffer. This will be bumped to + * successive rows with each change in the pitch angle. + */ + + buffer = &ginfo->swbuffer[IMAGE_TOP]; + + /* Loop through each row at each pitch angle */ + + for (row = 0; (row < (IMAGE_HEIGHT - VGULP_SIZE + 1)); row += VGULP_SIZE) + { + /* Initialize the pitch angles that will be needed in the inner loop. + * These are pre-calculated so that once we get started, we need not have + * to be concerned about zero crossing conditions. + */ + + g_pitch[0] = pitch; + g_buffer_row[0] = &buffer[IMAGE_LEFT]; + + for (i = 1; i < VGULP_SIZE; i++) + { + g_pitch[i] = g_pitch[i - 1] - VIDEO_ROW_ANGLE; + if (g_pitch[i] < ANGLE_0) + { + g_pitch[i] += ANGLE_360; + } + + g_buffer_row[i] = g_buffer_row[i - 1] + TRV_SCREEN_WIDTH; + } + + /* Seed the algorithm PART II: Set up the raycaster for this horizontal + * swathe + */ + + trv_ray_pitchprune(g_pitch[VGULP_SIZE - 1], g_pitch[0]); + + /* Seed the algorithm PART III: These initial hits will be moved to the + * beginning the hit array on the first pass through the loop. + */ + + trv_raycast(g_pitch[TOP_ROW], g_yaw[IMAGE_WIDTH], + RELYAW(IMAGE_WIDTH), &g_ray_hit[TOP_ROW][LEFT_COL]); + trv_raycast(g_pitch[BOT_ROW], g_yaw[IMAGE_WIDTH], + RELYAW(IMAGE_WIDTH), &g_ray_hit[BOT_ROW][LEFT_COL]); + + /* Loop through all columns at each yaw angle on the screen window */ + + for (g_cell_column = (IMAGE_WIDTH - HGULP_SIZE + 1); + g_cell_column >= 0; g_cell_column -= HGULP_SIZE) + { + trv_vdebug("\ng_cell_column=%d yaw=%d", g_cell_column, + g_yaw[g_cell_column]); + + /* Perform Ray VGULP_SIZE x HGULP_SIZE Casting */ + + /* The hits at the right corners will be the same as the hits for for + * the left hand corners on the next pass */ + + g_ray_hit[TOP_ROW][RIGHT_COL] = g_ray_hit[TOP_ROW][LEFT_COL]; + g_ray_hit[BOT_ROW][RIGHT_COL] = g_ray_hit[BOT_ROW][LEFT_COL]; + + /* Now get new hits in the right corners. */ + + trv_raycast(g_pitch[TOP_ROW], g_yaw[g_cell_column], + RELYAW(g_cell_column), &g_ray_hit[TOP_ROW][LEFT_COL]); + trv_raycast(g_pitch[BOT_ROW], g_yaw[g_cell_column], + RELYAW(g_cell_column), &g_ray_hit[BOT_ROW][LEFT_COL]); + + /* Now, resolve the cell recursively until the hits are the same in + * all four corners */ + + trv_resolve_cell(TOP_ROW, LEFT_COL, VGULP_SIZE, (HGULP_SIZE + 1)); + } + + /* End of the pitch loop. Bump up the pitch angle and the rending buffer + * pointer for the next time through the outer loop */ + + pitch -= (VGULP_SIZE * VIDEO_ROW_ANGLE); + if (pitch < ANGLE_0) + { + pitch += ANGLE_360; + } + + buffer += (VGULP_SIZE * TRV_SCREEN_WIDTH); + + /* Inform the ray cast engine that we are done with this horizonatal + * swathe. + */ + + trv_ray_pitchunprune(); + } + + /* Inform the ray cast engine that we are done. */ + + trv_ray_yawunprune(); +} + +/**************************************************************************** + * Function: trv_get_texture + * + * Description: + * This function returns the cell texture at the current cell row and + * column. This function is used by the texture rending functions when + * processing a cell on a transparent wall and a transparent pixel is + * encountered. This is a very inefficient way to handle these cases! + * Prevention of this condition is the best approach. However, total + * elimination of the condition is impossible. + * + ***************************************************************************/ + +uint8_t trv_get_texture(uint8_t row, uint8_t col) +{ + FAR struct trv_raycast_s *ptr = &g_ray_hit[row][col]; + FAR uint8_t *palptr; + int16_t zone; + + /* Perform a ray cast to get the hit at this row & column */ + + trv_raycast(g_pitch[row], g_yaw[g_cell_column + col], + RELYAW(g_cell_column + col), ptr); + + /* Check if we hit anything */ + + if (ptr->rect) + { + /* Get a pointer to the palette map to use on this pixel */ + + if (IS_SHADED(ptr->rect)) + { + zone = GET_ZONE(ptr->xdist, ptr->ydist); + palptr = GET_PALPTR(zone); + } + else + { + palptr = GET_PALPTR(0); + } + + /* We did, return the pixel at this location */ + /* PROBLEM: Need to know if the is an even or odd hit */ + + return palptr[GET_FRONT_PIXEL(ptr->rect, ptr->xpos, ptr->ypos)]; + } + else + { + return INVISIBLE_PIXEL; + } +} diff --git a/graphics/traveler/src/trv_rayprune.c b/graphics/traveler/src/trv_rayprune.c new file mode 100644 index 000000000..ce1bb7a83 --- /dev/null +++ b/graphics/traveler/src/trv_rayprune.c @@ -0,0 +1,1379 @@ +/******************************************************************************* + * apps/graphics/traveler/src/trv_rayprune.c + * + * Copyright (C) 2014 Gregory Nutt. All rights reserved. + * Author: Gregory Nutt + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in + * the documentation and/or other materials provided with the + * distribution. + * 3. Neither the name NuttX nor the names of its contributors may be + * used to endorse or promote products derived from this software + * without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, + * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, + * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS + * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED + * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN + * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + * + ****************************************************************************/ + +/**************************************************************************** + * Included files + ****************************************************************************/ + +#include "trv_types.h" +#include "trv_world.h" +#include "trv_plane.h" +#include "trv_trigtbl.h" +#include "trv_raycast.h" +#include "trv_rayprune.h" + +/**************************************************************************** + * Pre-processor Definitions + ****************************************************************************/ + +/* These definitions are used to decompose a range of angles into quadrants */ + +#define START0 0x0001 +#define STARTQ1 0x0002 +#define START90 0x0004 +#define STARTQ2 0x0008 +#define START180 0x0010 +#define STARTQ3 0x0020 +#define START270 0x0040 +#define STARTQ4 0x0080 +#define END0 0x0100 +#define ENDQ1 0x0200 +#define END90 0x0400 +#define ENDQ2 0x0800 +#define END180 0x1000 +#define ENDQ3 0x2000 +#define END270 0x4000 +#define ENDQ4 0x8000 + +#define S0EQ1 (START0|ENDQ1) +#define SQ1EQ1 (STARTQ1|ENDQ1) +#define SQ1E90 (STARTQ1|END90) +#define SQ1EQ2 (STARTQ1|ENDQ2) +#define S90EQ2 (START90|ENDQ2) +#define SQ2EQ2 (STARTQ2|ENDQ2) +#define SQ2E180 (STARTQ2|END180) +#define SQ2EQ3 (STARTQ2|ENDQ3) +#define S180EQ3 (START180|ENDQ3) +#define SQ3EQ3 (STARTQ3|ENDQ3) +#define SQ3E270 (STARTQ3|END270) +#define SQ3EQ4 (STARTQ3|ENDQ4) +#define S270EQ4 (START270|ENDQ4) +#define SQ4EQ4 (STARTQ4|ENDQ4) +#define SQ4E0 (STARTQ4|END0) +#define SQ4EQ1 (STARTQ4|ENDQ1) + +/**************************************************************************** + * Private Type Declarations + ***************************************************************************/ + +enum working_plane_state_e +{ + WEMPTY = 0, + WTOP, + WBOTTOM +}; + +/**************************************************************************** + * Private Function Prototypes + ****************************************************************************/ + +static void trv_ray_yawxprune_14cw(FAR struct trv_rect_head_s *outlist, + FAR struct trv_rect_head_s *inlist, + int16_t yaw); +static void trv_ray_yawxprune_14ccw(FAR struct trv_rect_head_s *outlist, + FAR struct trv_rect_head_s *inlist, + int16_t yaw); +static void trv_ray_yawxprune_23cw(FAR struct trv_rect_head_s *outlist, + FAR struct trv_rect_head_s *inlist, + int16_t yaw); +static void trv_ray_yawxprune_23ccw(FAR struct trv_rect_head_s *outlist, + FAR struct trv_rect_head_s *inlist, + int16_t yaw); +static void trv_ray_yawyprune_12cw(FAR struct trv_rect_head_s *outlist, + FAR struct trv_rect_head_s *inlist, + int16_t yaw); +static void trv_ray_yawyprune_12ccw(FAR struct trv_rect_head_s *outlist, + FAR struct trv_rect_head_s *inlist, + int16_t yaw); +static void trv_ray_yawyprune_34cw(FAR struct trv_rect_head_s *outlist, + FAR struct trv_rect_head_s *inlist, + int16_t yaw); +static void trv_ray_yawyprune_34ccw(FAR struct trv_rect_head_s *outlist, + FAR struct trv_rect_head_s *inlist, + int16_t yaw); +static void trv_ray_mkplanelist(FAR struct trv_rect_head_s *outlist, + FAR struct trv_rect_head_s *inlist, + int16_t yaw); +static void trv_ray_selectupper(FAR struct trv_rect_head_s *outlist, + FAR struct trv_rect_head_s *inlist); +static void trv_ray_selectlower(FAR struct trv_rect_head_s *outlist, + FAR struct trv_rect_head_s *inlist); + +/**************************************************************************** + * Public Data + ****************************************************************************/ + +struct trv_rect_head_s g_ray_xplane; /* List of X=plane rectangles */ +struct trv_rect_head_s g_ray_yplane; /* List of Y=plane rectangles */ +struct trv_rect_head_s g_ray_zplane; /* List of Z=plane rectangles */ + +/**************************************************************************** + * Private Data + ****************************************************************************/ + +static enum working_plane_state_e g_working_planestate; +static struct trv_rect_head_s g_working_xplane; /* List of X=plane rectangles */ +static struct trv_rect_head_s g_working_yplane; /* List of Y=plane rectangles */ + +static struct trv_rect_head_s g_candidate_xplane; /* List of X=plane rectangles */ +static struct trv_rect_head_s g_candidate_yplane; /* List of Y=plane rectangles */ +static struct trv_rect_head_s g_candidate_zplane; /* List of Z=plane rectangles */ + +static struct trv_rect_head_s g_discard_xplane; /* List of discarded X=plane + * rectangles */ +static struct trv_rect_head_s g_discard_yplane; /* List of discarded Y=plane + * rectangles */ +static struct trv_rect_head_s g_discard_zplane; /* List of discarded Z=plane + * rectangles */ + +/**************************************************************************** + * Private Functions + ****************************************************************************/ + +/**************************************************************************** + * Name: trv_ray_yawxprune_14cw + * + * Description: + * This function casts a ray along the Y-Axis looking at points of + * intersection on the X-Axis. We are pointed toward in the "right" half + * of the view. The X ray cast will proceed positively along the X axis. + * Every plane in the input plane list that lies to the "right" of the + * camera position and "clockwise" of the current yaw is moved into + * the output X plane list. + * + ***************************************************************************/ + +static void trv_ray_yawxprune_14cw(FAR struct trv_rect_head_s *outlist, + FAR struct trv_rect_head_s *inlist, + int16_t yaw) +{ + FAR struct trv_rect_list_s *entry; /* Points to the current X plane + * rectangle */ + FAR struct trv_rect_list_s *next; /* Points to the next X plane + * rectangle */ + FAR struct trv_rect_data_s *rect; /* Points to the rectangle data */ + trv_coord_t relx; /* Relative position of the X plane */ + trv_coord_t absy; /* Absolute Y position at relx given yaw */ + trv_coord_t lastrelx = -1; /* Last relative X position processed */ + int32_t dydx; /* Rate of change of Y wrt X (double) */ + + /* Pre-calculate the rate of change of Y with respect to X */ + /* The tangent is equal to the rate of change of Y with respect to the + * X-axis. The tangent is stored at double the "normal" scaling. + */ + + dydx = TAN(yaw); + + /* Look at every rectangle lying in the X plane */ + /* This logic should be improved at some point so that non-visible planes + * are "pruned" from the list prior to ray casting! + */ + + for (entry = inlist->head; (entry); entry = next) + { + next = entry->flink; + rect = &entry->d; + + /* Search for a rectangle which lies "beyond" the current camera + * position + */ + + if (rect->plane > g_camera.x) + { + /* Get the X distance to the plane */ + + relx = rect->plane - g_camera.x; + + /* Calculate the Y position at this relative X position. We can + * skip this step if we are processing another rectangle at the + * same relx distance. + */ + + if (relx != lastrelx) + { + int32_t deltay; /* Scale == "triple" */ + + /* The dydx is stored at double the"normal" scaling -- so deltay + * is "triple" precision + */ + + deltay = dydx * ((int32_t) relx); + absy = tTOs(deltay) + g_camera.y; /* back to "single" */ + lastrelx = relx; + } + + /* Check if any part of this plane lies clockwise from this Y + * position. In quadrants 1 & 4, clockwise corresponds to smaller Y + * values + */ + + if (absy >= rect->hstart) + { + /* Remove the plane from the input plane list and add it to the + * output plane list + */ + + trv_move_plane(entry, outlist, inlist); + } + } + } +} + +/**************************************************************************** + * Name: trv_ray_yawxprune_14ccw + * + * Description: + * This function casts a ray along the Y-Axis looking at points of + * intersection on the X-Axis. We are pointed toward in the "right" half + * of the view. The X ray cast will proceed positively along the X axis. + * Every plane in the input plane list that lies to the "right" of the + * camera position and "counterclockwise" of the current yaw is moved into + * the output X plane list. + * + ***************************************************************************/ + +static void trv_ray_yawxprune_14ccw(FAR struct trv_rect_head_s *outlist, + FAR struct trv_rect_head_s *inlist, + int16_t yaw) +{ + FAR struct trv_rect_list_s *entry; /* Points to the current X plane + * rectangle */ + FAR struct trv_rect_list_s *next; /* Points to the next X plane + * rectangle */ + FAR struct trv_rect_data_s *rect; /* Points to the rectangle data */ + trv_coord_t relx; /* Relative position of the X plane */ + trv_coord_t absy; /* Absolute Y position at relx given yaw */ + trv_coord_t lastrelx = -1; /* Last relative X position processed */ + int32_t dydx; /* Rate of change of Y wrt X (double) */ + + /* Pre-calculate the rate of change of Y with respect to X */ + /* The tangent is equal to the rate of change of Y with respect to the + * X-axis. The tangent is stored at double the "normal" scaling. + */ + + dydx = TAN(yaw); + + /* Look at every rectangle lying in the X plane */ + /* This logic should be improved at some point so that non-visible planes + * are "pruned" from the list prior to ray casting! + */ + + for (entry = inlist->head; (entry); entry = next) + { + next = entry->flink; + rect = &entry->d; + + /* Search for a rectangle which lies "beyond" the current camera + * position + */ + + if (rect->plane > g_camera.x) + { + /* Get the X distance to the plane */ + + relx = rect->plane - g_camera.x; + + /* Calculate the Y position at this relative X position. We can + * skip this step if we are processing another rectangle at the + * same relx distance. + */ + + if (relx != lastrelx) + { + int32_t deltay; /* Scale == "triple" */ + + /* The dydx is stored at double the"normal" scaling -- so deltay + * is "triple" precision + */ + + deltay = dydx * ((int32_t) relx); + absy = tTOs(deltay) + g_camera.y; /* back to "single" */ + lastrelx = relx; + } + + /* Check if any part of this plane lies counterclockwise from this Y + * position. In quadrants 1 & 4, counterclockwise corresponds to + * larger Y values + */ + + if (absy <= rect->hend) + { + /* Remove the plane from the input plane list and add it to the + * output plane list + */ + + trv_move_plane(entry, outlist, inlist); + } + } + } +} + +/**************************************************************************** + * Name: rayYawYPrune23CW + * + * Description: + * This function casts a ray along the Y-Axis looking at points of + * intersection on the X-Axis. We are pointed toward in the "left" half + * of the view. The X ray cast will proceed negatively along the X axis. + * Every plane in the input X plane list that lies to the "left" of the + * camera position and "clockwise" of the current yaw is moved into + * the output X plane list. + * + ***************************************************************************/ + +static void trv_ray_yawxprune_23cw(FAR struct trv_rect_head_s *outlist, + FAR struct trv_rect_head_s *inlist, + int16_t yaw) +{ + FAR struct trv_rect_list_s *entry; /* Points to the current X plane + * rectangle */ + FAR struct trv_rect_list_s *next; /* Points to the next X plane + * rectangle */ + FAR struct trv_rect_data_s *rect; /* Points to the rectangle data */ + trv_coord_t relx; /* Relative position of the X plane */ + trv_coord_t absy; /* Absolute Y position at relx given yaw */ + trv_coord_t lastrelx = -1; /* Last relative X position processed */ + int32_t dydx; /* Rate of change of Y wrt X (double) */ + + /* Pre-calculate the rate of change of Y with respect to X */ + /* The negative tangent is equal to the rate of change of Y with respect + * to the X-axis. The tangent is stored at double the "normal" scaling. + */ + + dydx = -TAN(yaw); + + /* Look at every rectangle lying in the X plane */ + /* This logic should be improved at some point so that non-visible planes + * are "pruned" from the list prior to ray casting! + */ + + for (entry = inlist->tail; (entry); entry = next) + { + next = entry->blink; + rect = &entry->d; + + /* Search for a rectangle which lies "before" the current camera + * position + */ + + if (rect->plane < g_camera.x) + { + /* Get the X distance to the plane */ + + relx = g_camera.x - rect->plane; + + /* Calculate the Y position at this relative X position. We can + * skip this step if we are processing another rectangle at the + * same relx distance. + */ + + if (relx != lastrelx) + { + int32_t deltay; /* Scale == "triple" */ + + /* The dydx is stored at double the"normal" scaling -- so deltay + * is "triple" precision + */ + + deltay = dydx * ((int32_t) relx); + absy = tTOs(deltay) + g_camera.y; /* back to "single" */ + lastrelx = relx; + } + + /* Check if any part of this plane lies clockwise from this Y + * position. In quadrants 2 & 3, clockwise corresponds to larger Y + * values + */ + + if (absy <= rect->hend) + { + /* Remove the plane from the input plane list and add it to the + * output plane list + */ + + trv_move_plane(entry, outlist, inlist); + } + } + } +} + +/**************************************************************************** + * Name: rayYawYPrune23CCW + * + * Description: + * This function casts a ray along the Y-Axis looking at points of + * intersection on the X-Axis. We are pointed toward in the "left" half + * of the view. The X ray cast will proceed negatively along the X axis. + * Every plane in the input X plane list that lies to the "left" of the + * camera position and "counterclockwise" of the current yaw is moved + * into the output X plane list. + * + ***************************************************************************/ + +static void trv_ray_yawxprune_23ccw(FAR struct trv_rect_head_s *outlist, + FAR struct trv_rect_head_s *inlist, + int16_t yaw) +{ + FAR struct trv_rect_list_s *entry; /* Points to the current X plane + * rectangle */ + FAR struct trv_rect_list_s *next; /* Points to the next X plane + * rectangle */ + FAR struct trv_rect_data_s *rect; /* Points to the rectangle data */ + trv_coord_t relx; /* Relative position of the X plane */ + trv_coord_t absy; /* Absolute Y position at relx given yaw */ + trv_coord_t lastrelx = -1; /* Last relative X position processed */ + int32_t dydx; /* Rate of change of Y wrt X (double) */ + + /* Pre-calculate the rate of change of Y with respect to X */ + /* The negative tangent is equal to the rate of change of Y with respect + * to the X-axis. The tangent is stored at double the "normal" scaling. + */ + + dydx = -TAN(yaw); + + /* Look at every rectangle lying in the X plane */ + /* This logic should be improved at some point so that non-visible planes + * are "pruned" from the list prior to ray casting! + */ + + for (entry = inlist->tail; (entry); entry = next) + { + next = entry->blink; + rect = &entry->d; + + /* Search for a rectangle which lies "before" the current camera + * position + */ + + if (rect->plane < g_camera.x) + { + /* Get the X distance to the plane */ + + relx = g_camera.x - rect->plane; + + /* Calculate the Y position at this relative X position. We can + * skip this step if we are processing another rectangle at the + * same relx distance. + */ + + if (relx != lastrelx) + { + int32_t deltay; /* Scale == "triple" */ + + /* The dydx is stored at double the"normal" scaling -- so deltay + * is "triple" precision + */ + + deltay = dydx * ((int32_t) relx); + absy = tTOs(deltay) + g_camera.y; /* back to "single" */ + lastrelx = relx; + } + + /* Check if any part of this plane lies counterclockwise from this Y + * position. In quadrants 2 & 3, counterclockwise corresponds to + * smaller Y values + */ + + if (absy >= rect->hstart) + { + /* Remove the plane from the input plane list and add it to the + * output plane list + */ + + trv_move_plane(entry, outlist, inlist); + } + } + } +} + +/**************************************************************************** + * Name: trv_ray_yawyprune_12cw + * + * Description: + * This function casts a ray along the X-Axis looking at points of + * intersection on the Y-Axis. We are pointed toward in the "forward" half + * of the view. The Y ray cast will proceed positively along the Y axis. + * Every plane in the input plane list that lies to the "forward" of the + * camera position and "clockwise" of the current yaw is moved into + * the output Y plane list. + * + ***************************************************************************/ + +static void trv_ray_yawyprune_12cw(FAR struct trv_rect_head_s *outlist, + FAR struct trv_rect_head_s *inlist, + int16_t yaw) +{ + FAR struct trv_rect_list_s *entry; /* Points to the current Y plane + * rectangle */ + FAR struct trv_rect_list_s *next; /* Points to the next Y plane + * rectangle */ + FAR struct trv_rect_data_s *rect; /* Points to the rectangle data */ + trv_coord_t rely; /* Relative position of the Y plane */ + trv_coord_t absx; /* Absolute X position at rely given yaw */ + trv_coord_t lastrely = -1; /* Last relative Y position processed */ + int32_t dxdy; /* Rate of change of X wrt Y (double) */ + + /* Pre-calculate the rate of change of X with respect to Y */ + /* The inverted tangent is equal to the rate of change of X with respect + * to the Y-axis. The cotangent is stored at double the the "normal" + * scaling. + */ + + dxdy = g_cot_table(yaw); + + /* Look at every rectangle lying in the Y plane */ + /* This logic should be improved at some point so that non-visible planes + * are "pruned" from the list prior to ray casting! + */ + + for (entry = inlist->head; (entry); entry = next) + { + next = entry->flink; + rect = &entry->d; + + /* Search for a rectangle which lies "beyond" the current camera + * position + */ + + if (rect->plane > g_camera.y) + { + /* Get the Y distance to the plane */ + + rely = rect->plane - g_camera.y; + + /* Calculate the X position at this relative Y position. We can + * skip this step if we are processing another rectangle at the + * same rely distance. + */ + + if (rely != lastrely) + { + int32_t deltax; /* Scale == "triple" */ + + /* The dxdy is stored at double the"normal" scaling -- so deltax + * is "triple" precision + */ + + deltax = dxdy * ((int32_t) rely); + absx = tTOs(deltax) + g_camera.x; /* back to "single" */ + lastrely = rely; + } + + /* Check if any part of this plane lies clockwise from this X + * position. In quadrants 1 & 2, clockwise corresponds to larger + * X values + */ + + if (absx <= rect->hend) + { + + /* Remove the plane from the input plane list and add it to the + * output plane list + */ + + trv_move_plane(entry, outlist, inlist); + } + } + } +} + +/**************************************************************************** + * Name: trv_ray_yawyprune_12ccw + * + * Description: + * This function casts a ray along the X-Axis looking at points of + * intersection on the Y-Axis. We are pointed toward in the "forward" half + * of the view. The Y ray cast will proceed positively along the Y axis. + * Every plane in the input plane list that lies to the "forward" of the + * camera position and "counterclockwise" of the current yaw is moved into + * the output Y plane list. + * + ***************************************************************************/ + +static void trv_ray_yawyprune_12ccw(FAR struct trv_rect_head_s *outlist, + FAR struct trv_rect_head_s *inlist, + int16_t yaw) +{ + FAR struct trv_rect_list_s *entry; /* Points to the current Y plane + * rectangle */ + FAR struct trv_rect_list_s *next; /* Points to the next Y plane + * rectangle */ + FAR struct trv_rect_data_s *rect; /* Points to the rectangle data */ + trv_coord_t rely; /* Relative position of the Y plane */ + trv_coord_t absx; /* Absolute X position at rely given yaw */ + trv_coord_t lastrely = -1; /* Last relative Y position processed */ + int32_t dxdy; /* Rate of change of X wrt Y (double) */ + + /* Pre-calculate the rate of change of X with respect to Y */ + /* The inverted tangent is equal to the rate of change of X with respect + * to the Y-axis. The cotangent is stored at double the the "normal" + * scaling. + */ + + dxdy = g_cot_table(yaw); + + /* Look at every rectangle lying in the Y plane */ + /* This logic should be improved at some point so that non-visible planes + * are "pruned" from the list prior to ray casting! + */ + + for (entry = inlist->head; (entry); entry = next) + { + next = entry->flink; + rect = &entry->d; + + /* Search for a rectangle which lies "beyond" the current camera + * position + */ + + if (rect->plane > g_camera.y) + { + /* Get the Y distance to the plane */ + + rely = rect->plane - g_camera.y; + + /* Calculate the X position at this relative Y position. We can + * skip this step if we are processing another rectangle at the + * same rely distance. + */ + + if (rely != lastrely) + { + int32_t deltax; /* Scale == "triple" */ + + /* The dxdy is stored at double the"normal" scaling -- so + * deltax is "triple" precision + */ + + deltax = dxdy * ((int32_t) rely); + absx = tTOs(deltax) + g_camera.x; /* back to "single" */ + lastrely = rely; + } + + /* Check if any part of this plane lies counterclockwise from this + * X position. In quadrants 1 & 2, counterclockwise corresponds + * to smaller X values + */ + + if (absx >= rect->hstart) + { + + /* Remove the plane from the input plane list and add it to the + * output plane list + */ + + trv_move_plane(entry, outlist, inlist); + } + } + } +} + +/**************************************************************************** + * Name: trv_ray_yawyprune_34cw + * + * Description: + * This function casts a ray along the X-Axis looking at points of + * intersection on the Y-Axis. We are pointed toward in the "back" half + * of the view. The Y ray cast will proceed negatively along the Y axis. + * Every plane in the input plane list that lies to the "back" of the + * camera position and "clockwise" of the current yaw is moved into + * the output Y plane list. + * + ***************************************************************************/ + +static void trv_ray_yawyprune_34cw(FAR struct trv_rect_head_s *outlist, + FAR struct trv_rect_head_s *inlist, + int16_t yaw) +{ + FAR struct trv_rect_list_s *entry; /* Points to the current Y plane + * rectangle */ + FAR struct trv_rect_list_s *next; /* Points to the next Y plane + * rectangle */ + FAR struct trv_rect_data_s *rect; /* Points to the rectangle data */ + trv_coord_t rely; /* Relative position of the Y plane */ + trv_coord_t absx; /* Absolute X position at rely given yaw */ + trv_coord_t lastrely = -1; /* Last relative Y position processed */ + int32_t dxdy; /* Rate of change of X wrt Y (double) */ + + /* Pre-calculate the rate of change of X with respect to Y */ + /* The negative inverted tangent is equal to the rate of change of X with + * respect to the Y-axis. The cotangent is stored at double the the + * "normal" scaling. + */ + + dxdy = -g_cot_table(yaw - ANGLE_180); + + /* Look at every rectangle lying in the Y plane */ + /* This logic should be improved at some point so that non-visible planes + * are "pruned" from the list prior to ray casting! + */ + + for (entry = inlist->tail; (entry); entry = next) + { + next = entry->blink; + rect = &entry->d; + + /* Search for a rectangle which lies "before" the current camera + * position + */ + + if (rect->plane < g_camera.y) + { + /* Get the Y distance to the plane */ + + rely = g_camera.y - rect->plane; + + /* Calculate the X position at this relative Y position. We can skip + * this step if we are processing another rectangle at the same rely + * distance. + */ + + if (rely != lastrely) + { + int32_t deltax; /* Scale == "triple" */ + + /* The dxdy is stored at double the"normal" scaling -- so + * deltax is "triple" precision + */ + + deltax = dxdy * ((int32_t) rely); + absx = tTOs(deltax) + g_camera.x; /* back to "single" */ + lastrely = rely; + } + + /* Check if any part of this plane lies clockwise from this X + * position. In quadrants 3 & 4, clockwise corresponds to + * smaller X values + */ + + if (absx >= rect->hstart) + { + + /* Remove the plane from the input plane list and add it to + * the output plane list + */ + + trv_move_plane(entry, outlist, inlist); + } + } + } +} + +/**************************************************************************** + * Name: trv_ray_yawyprune_34ccw + * + * Description: + * This function casts a ray along the X-Axis looking at points of + * intersection on the Y-Axis. We are pointed toward in the "back" half + * of the view. The Y ray cast will proceed negatively along the Y axis. + * Every plane in the input plane list that lies to the "back" of the + * camera position and "counterclockwise" of the current yaw is moved + * into the output Y plane list. + * + ***************************************************************************/ + +static void trv_ray_yawyprune_34ccw(FAR struct trv_rect_head_s *outlist, + FAR struct trv_rect_head_s *inlist, + int16_t yaw) +{ + FAR struct trv_rect_list_s *entry; /* Points to the current Y plane + * rectangle */ + FAR struct trv_rect_list_s *next; /* Points to the next Y plane + * rectangle */ + FAR struct trv_rect_data_s *rect; /* Points to the rectangle data */ + trv_coord_t rely; /* Relative position of the Y plane */ + trv_coord_t absx; /* Absolute X position at rely given yaw */ + trv_coord_t lastrely = -1; /* Last relative Y position processed */ + int32_t dxdy; /* Rate of change of X wrt Y (double) */ + + /* Pre-calculate the rate of change of X with respect to Y */ + /* The negative inverted tangent is equal to the rate of change of X with + * respect to the Y-axis. The cotangent is stored at double the the + * "normal" scaling. + */ + + dxdy = -g_cot_table(yaw - ANGLE_180); + + /* Look at every rectangle lying in the Y plane */ + /* This logic should be improved at some point so that non-visible planes + * are "pruned" from the list prior to ray casting! + */ + + for (entry = inlist->tail; (entry); entry = next) + { + next = entry->blink; + rect = &entry->d; + + /* Search for a rectangle which lies "before" the current camera + * position + */ + + if (rect->plane < g_camera.y) + { + /* Get the Y distance to the plane */ + + rely = g_camera.y - rect->plane; + + /* Calculate the X position at this relative Y position. We can + * skip this step if we are processing another rectangle at the + * same rely distance. + */ + + if (rely != lastrely) + { + int32_t deltax; /* Scale == "triple" */ + + /* The dxdy is stored at double the"normal" scaling -- so + * deltax is "triple" precision + */ + + deltax = dxdy * ((int32_t) rely); + absx = tTOs(deltax) + g_camera.x; /* back to "single" */ + lastrely = rely; + } + + /* Check if any part of this plane lies counterclockwise from this X + * position. In quadrants 3 & 4, counterclockwise corresponds to + * larger X values + */ + + if (absx <= rect->hend) + { + /* Remove the plane from the input plane list and add it to the + * output plane list + */ + + trv_move_plane(entry, outlist, inlist); + } + } + } +} + +/**************************************************************************** + * Name: trv_ray_mkplanelist + * + * Description: + * This function creates a "fake" plane list based on the current working + * plane list. The "head" of the "fake" plane list points to the first + * entry "beyond" the current camera position; the "tail" point to the + * first plane "before" the current position. + * + ****************************************************************************/ + +static void trv_ray_mkplanelist(FAR struct trv_rect_head_s *outlist, + FAR struct trv_rect_head_s *inlist, + int16_t pos) +{ + FAR struct trv_rect_list_s *list; /* Points to the current plane rectangle */ + + /* Look at every rectangle lying in the specified "in" plane starting at + * the beginning of the list. + */ + + outlist->head = NULL; + for (list = inlist->head; list; list = list->flink) + { + /* Search for the first rectangle which lies "beyond" the current + * camera position + */ + + if (list->d.plane > pos) + { + outlist->head = list; + break; + } + } + + /* Look at every rectangle lying in the specified "in" plane starting at + * the end of the list + */ + + outlist->tail = NULL; + for (list = inlist->tail; list; list = list->blink) + { + /* Search for the first rectangle which lies "before" the current + * camera position + */ + + if (list->d.plane < pos) + { + outlist->tail = list; + break; + } + } +} + +/**************************************************************************** + * Name: trv_ray_selectupper + * + * Description: + * Transfer each plane that lies (at least in part) above the level viewing + * angle from the inlist to the outlist). + * + ****************************************************************************/ + +static void trv_ray_selectupper(FAR struct trv_rect_head_s *outlist, + FAR struct trv_rect_head_s *inlist) +{ + FAR struct trv_rect_list_s *entry; /* Points to the current Y plane + * rectangle */ + FAR struct trv_rect_list_s *next; /* Points to the next Y plane + * rectangle */ + + /* Look at every rectangle lying in the inlist */ + + for (entry = inlist->head; (entry); entry = next) + { + next = entry->flink; + + /* If the "top" of the rectangle lies above the camera Z level, then + * some (or all) of the rectangle is above the level viewing angle. + */ + + if (entry->d.vend >= g_camera.z) + { + /* Remove the plane from the input plane list and add it to the + * output plane list + */ + + trv_move_plane(entry, outlist, inlist); + } + } +} + +/**************************************************************************** + * Name: trv_ray_selectlower + * + * Description: + * Transfer each plane that lies (at least in part) above the level viewing + * angle from the inlist to the outlist). + * + ****************************************************************************/ + +static void trv_ray_selectlower(FAR struct trv_rect_head_s *outlist, + FAR struct trv_rect_head_s *inlist) +{ + FAR struct trv_rect_list_s *entry; /* Points to the current Y plane + * rectangle */ + FAR struct trv_rect_list_s *next; /* Points to the next Y plane + * rectangle */ + + /* Look at every rectangle lying in the inlist */ + + for (entry = inlist->head; (entry); entry = next) + { + next = entry->flink; + + /* If the "bottom" of the rectangle lies below the camera Z level, then + * some (or all) of the rectangle is above the level viewing angle. + */ + + if (entry->d.vstart < g_camera.z) + { + /* Remove the plane from the input plane list and add it to the + * output plane list + */ + + trv_move_plane(entry, outlist, inlist); + } + } +} + +/**************************************************************************** + * Public Functions + ****************************************************************************/ + +/**************************************************************************** + * Name: trv_ray_yawprune + * + * Description: + * This function may be called at the beginning of each rending cycle to + * prepare ray casting data. This function "prunes" out all planes that + * may be visible in the specified yaw range. + * + ****************************************************************************/ + +void trv_ray_yawprune(int16_t yawstart, int16_t yawend) +{ + uint16_t quadrants; + + /* Nullify the working plane lists */ + + g_working_xplane.head = g_working_xplane.tail = NULL; + g_working_xplane.head = g_working_xplane.tail = NULL; + g_working_planestate = WEMPTY; + + /* Nullify the candidate plane lists */ + + g_candidate_xplane.head = g_candidate_xplane.tail = NULL; + g_candidate_yplane.head = g_candidate_yplane.tail = NULL; + g_candidate_zplane.head = g_candidate_zplane.tail = NULL; + + /* Nullify the discard plane lists */ + + g_discard_xplane.head = g_discard_xplane.tail = NULL; + g_discard_yplane.head = g_discard_yplane.tail = NULL; + g_discard_zplane.head = g_discard_zplane.tail = NULL; + + /* Determine the quadrant(s) spanned by yawstart and yawend */ + + if (yawstart == ANGLE_0) + { + quadrants = START0; + } + else if (yawstart < ANGLE_90) + { + quadrants = STARTQ1; + } + else if (yawstart == ANGLE_90) + { + quadrants = START90; + } + else if (yawstart < ANGLE_180) + { + quadrants = STARTQ2; + } + else if (yawstart == ANGLE_180) + { + quadrants = START180; + } + else if (yawstart < ANGLE_270) + { + quadrants = STARTQ3; + } + else if (yawstart == ANGLE_270) + { + quadrants = START270; + } + else + { + quadrants = STARTQ4; + } + + if (yawend == ANGLE_0) + { + quadrants |= END0; + } + else if (yawend < ANGLE_90) + { + quadrants |= ENDQ1; + } + else if (yawend == ANGLE_90) + { + quadrants |= END90; + } + else if (yawend < ANGLE_180) + { + quadrants |= ENDQ2; + } + else if (yawend == ANGLE_180) + { + quadrants |= END180; + } + else if (yawend < ANGLE_270) + { + quadrants |= ENDQ3; + } + else if (yawend == ANGLE_270) + { + quadrants |= END270; + } + else + { + quadrants |= ENDQ4; + } + + /* Now, perform the pruning based upon the quadrant(s) spanned by yawstart + * and yawend + */ + + switch (quadrants) + { + case S0EQ1: + trv_ray_yawxprune_14ccw(&g_discard_xplane, &g_xplane, yawstart); + trv_ray_yawxprune_14cw(&g_candidate_xplane, &g_discard_xplane, yawend); + trv_ray_yawyprune_12cw(&g_candidate_yplane, &g_yplane, yawend); + break; + + case SQ1EQ1: + trv_ray_yawxprune_14ccw(&g_discard_xplane, &g_xplane, yawstart); + trv_ray_yawxprune_14cw(&g_candidate_xplane, &g_discard_xplane, yawend); + trv_ray_yawyprune_12ccw(&g_discard_yplane, &g_yplane, yawstart); + trv_ray_yawyprune_12cw(&g_candidate_yplane, &g_discard_yplane, yawend); + break; + + case SQ1E90: + trv_ray_yawxprune_14ccw(&g_candidate_xplane, &g_xplane, yawstart); + trv_ray_yawyprune_12ccw(&g_discard_yplane, &g_yplane, yawstart); + trv_ray_yawyprune_12cw(&g_candidate_yplane, &g_discard_yplane, yawend); + break; + + case SQ1EQ2: + trv_ray_yawxprune_14ccw(&g_discard_xplane, &g_xplane, yawstart); + trv_ray_yawxprune_23cw(&g_candidate_xplane, &g_xplane, yawend); + trv_merge_planelists(&g_candidate_xplane, &g_discard_xplane); + trv_ray_yawyprune_12ccw(&g_discard_yplane, &g_yplane, yawstart); + trv_ray_yawyprune_12cw(&g_candidate_yplane, &g_discard_yplane, yawend); + break; + + case S90EQ2: + trv_ray_yawxprune_23cw(&g_candidate_xplane, &g_xplane, yawend); + trv_ray_yawyprune_12ccw(&g_discard_yplane, &g_yplane, yawstart); + trv_ray_yawyprune_12cw(&g_candidate_yplane, &g_discard_yplane, yawend); + break; + + case SQ2EQ2: + trv_ray_yawxprune_23ccw(&g_discard_xplane, &g_xplane, yawstart); + trv_ray_yawxprune_23cw(&g_candidate_xplane, &g_discard_xplane, yawend); + trv_ray_yawyprune_12ccw(&g_discard_yplane, &g_yplane, yawstart); + trv_ray_yawyprune_12cw(&g_candidate_yplane, &g_discard_yplane, yawend); + break; + + case SQ2E180: + trv_ray_yawxprune_23ccw(&g_discard_xplane, &g_xplane, yawstart); + trv_ray_yawxprune_23cw(&g_candidate_xplane, &g_discard_xplane, yawend); + trv_ray_yawyprune_12ccw(&g_candidate_yplane, &g_yplane, yawstart); + break; + + case SQ2EQ3: + trv_ray_yawxprune_23ccw(&g_discard_xplane, &g_xplane, yawstart); + trv_ray_yawxprune_23cw(&g_candidate_xplane, &g_discard_xplane, yawend); + trv_ray_yawyprune_12ccw(&g_discard_yplane, &g_yplane, yawstart); + trv_ray_yawyprune_34cw(&g_candidate_yplane, &g_yplane, yawend); + trv_merge_planelists(&g_candidate_yplane, &g_discard_yplane); + break; + + case S180EQ3: + trv_ray_yawxprune_23ccw(&g_discard_xplane, &g_xplane, yawstart); + trv_ray_yawxprune_23cw(&g_candidate_xplane, &g_discard_xplane, yawend); + trv_ray_yawyprune_34cw(&g_candidate_yplane, &g_yplane, yawend); + break; + + case SQ3EQ3: + trv_ray_yawxprune_23ccw(&g_discard_xplane, &g_xplane, yawstart); + trv_ray_yawxprune_23cw(&g_candidate_xplane, &g_discard_xplane, yawend); + trv_ray_yawyprune_34ccw(&g_discard_yplane, &g_yplane, yawstart); + trv_ray_yawyprune_34cw(&g_candidate_yplane, &g_discard_yplane, yawend); + break; + + case SQ3E270: + trv_ray_yawxprune_23ccw(&g_candidate_xplane, &g_xplane, yawend); + trv_ray_yawyprune_34ccw(&g_discard_yplane, &g_yplane, yawstart); + trv_ray_yawyprune_34cw(&g_candidate_yplane, &g_discard_yplane, yawend); + break; + + case SQ3EQ4: + trv_ray_yawxprune_23ccw(&g_discard_xplane, &g_xplane, yawstart); + trv_ray_yawxprune_14cw(&g_candidate_xplane, &g_xplane, yawend); + trv_merge_planelists(&g_candidate_xplane, &g_discard_xplane); + trv_ray_yawyprune_34ccw(&g_discard_yplane, &g_yplane, yawstart); + trv_ray_yawyprune_34cw(&g_candidate_yplane, &g_discard_yplane, yawend); + break; + + case S270EQ4: + trv_ray_yawxprune_14cw(&g_candidate_xplane, &g_xplane, yawend); + trv_ray_yawyprune_34ccw(&g_discard_yplane, &g_yplane, yawstart); + trv_ray_yawyprune_34cw(&g_candidate_yplane, &g_discard_yplane, yawend); + break; + + case SQ4EQ4: + trv_ray_yawxprune_14ccw(&g_discard_xplane, &g_xplane, yawstart); + trv_ray_yawxprune_14cw(&g_candidate_xplane, &g_discard_xplane, yawend); + trv_ray_yawyprune_34ccw(&g_discard_yplane, &g_yplane, yawstart); + trv_ray_yawyprune_34cw(&g_candidate_yplane, &g_discard_yplane, yawend); + break; + + case SQ4E0: + trv_ray_yawxprune_14ccw(&g_discard_xplane, &g_xplane, yawstart); + trv_ray_yawxprune_14cw(&g_candidate_xplane, &g_discard_xplane, yawend); + trv_ray_yawyprune_34ccw(&g_candidate_yplane, &g_yplane, yawstart); + break; + + case SQ4EQ1: + trv_ray_yawxprune_14ccw(&g_discard_xplane, &g_xplane, yawstart); + trv_ray_yawxprune_14cw(&g_candidate_xplane, &g_discard_xplane, yawend); + trv_ray_yawyprune_34ccw(&g_discard_yplane, &g_yplane, yawstart); + trv_ray_yawyprune_12cw(&g_candidate_yplane, &g_yplane, yawend); + trv_merge_planelists(&g_candidate_yplane, &g_discard_yplane); + break; + + default: /* No other combination should occur if the range is < 90 deg */ + break; + } + + /* Create the special plane lists for the raycaster */ + + trv_ray_mkplanelist(&g_ray_xplane, &g_candidate_xplane, g_camera.x); + trv_ray_mkplanelist(&g_ray_yplane, &g_candidate_yplane, g_camera.y); + trv_ray_mkplanelist(&g_ray_zplane, &g_zplane, g_camera.z); +} + +/**************************************************************************** + * Name: trv_ray_pitchprune + * + * Description: + * This function may be called at each "horizontal" pitch swathe to + * prepare ray casting data. This function "prunes" out all planes that + * may be visible in the specified pitch range. + * + ****************************************************************************/ + +void trv_ray_pitchprune(int16_t pitchstart, int16_t pitchend) +{ + /* Check the current pitch range */ + + if (pitchend <= ANGLE_180) + { + /* The entire view is in the upper hemisphere */ + + if (g_working_planestate != WTOP) + { + if (g_working_planestate == WBOTTOM) + { + /* Merge the workingPlane lists with the candidate plane lists */ + + trv_merge_planelists(&g_candidate_xplane, &g_working_xplane); + trv_merge_planelists(&g_candidate_yplane, &g_working_yplane); + } + + /* Select the rectangles which lie above the level line of sight */ + + trv_ray_selectupper(&g_working_xplane, &g_candidate_xplane); + trv_ray_selectupper(&g_working_yplane, &g_candidate_yplane); + + /* Create the special plane lists for the raycaster */ + + trv_ray_mkplanelist(&g_ray_xplane, &g_working_xplane, g_camera.x); + trv_ray_mkplanelist(&g_ray_yplane, &g_working_yplane, g_camera.y); + + g_working_planestate = WTOP; + } + } + else if (pitchstart <= ANGLE_180) + { + /* The view is split between the upper and lower hemispheres */ + + if (g_working_planestate != WEMPTY) + { + /* Merge the workingPlane lists with the candidate plane lists */ + + trv_merge_planelists(&g_candidate_xplane, &g_working_xplane); + trv_merge_planelists(&g_candidate_yplane, &g_working_yplane); + + /* Create the special plane lists for the raycaster */ + + trv_ray_mkplanelist(&g_ray_xplane, &g_candidate_xplane, g_camera.x); + trv_ray_mkplanelist(&g_ray_yplane, &g_candidate_yplane, g_camera.y); + + g_working_planestate = WEMPTY; + } + } + else + { + /* The entire view is in the lower hemisphere */ + + if (g_working_planestate != WBOTTOM) + { + if (g_working_planestate == WTOP) + { + /* Merge the workingPlane lists with the candidate plane lists */ + + trv_merge_planelists(&g_candidate_xplane, &g_working_xplane); + trv_merge_planelists(&g_candidate_yplane, &g_working_yplane); + } + + /* Select the rectangles which lie below the level line of sight */ + + trv_ray_selectlower(&g_working_xplane, &g_candidate_xplane); + trv_ray_selectlower(&g_working_yplane, &g_candidate_yplane); + + /* Create the special plane lists for the raycaster */ + + trv_ray_mkplanelist(&g_ray_xplane, &g_working_xplane, g_camera.x); + trv_ray_mkplanelist(&g_ray_yplane, &g_working_yplane, g_camera.y); + + g_working_planestate = WBOTTOM; + } + } +} + +/**************************************************************************** + * Name: trv_ray_pitchunprune + * + * Description: + * This function may be called at the end of each rending cycle to + * restore the environment which was perturbed by trv_ray_pitchprune. + * + ****************************************************************************/ + +void trv_ray_pitchunprune(void) +{ +} + +/**************************************************************************** + * Name: trv_ray_yawunprune + * + * Description: + * This function may be called at the end of each rending cycle to + * restore the environment which was perturbed by trv_ray_yawprune. + * + ****************************************************************************/ + +void trv_ray_yawunprune(void) +{ + /* Merge the working X plane list with the discarded X plane list */ + + trv_merge_planelists(&g_discard_xplane, &g_working_xplane); + + /* Merge the candidate X plane list with the discarded X plane list */ + + trv_merge_planelists(&g_discard_xplane, &g_candidate_xplane); + + /* Merge the discarded X plane list with the original "raw" X plane list */ + + trv_merge_planelists(&g_xplane, &g_discard_xplane); + + /* Merge the working Y plane list with the discarded Y plane list */ + + trv_merge_planelists(&g_discard_yplane, &g_working_yplane); + + /* Merge the candidate X plane list with the discarded X plane list */ + + trv_merge_planelists(&g_discard_yplane, &g_candidate_yplane); + + /* Merge the discarded Y plane list with the original "raw" Y plane list */ + + trv_merge_planelists(&g_yplane, &g_discard_yplane); +} diff --git a/graphics/traveler/src/trv_rayrend.c b/graphics/traveler/src/trv_rayrend.c new file mode 100644 index 000000000..f9c45c3ad --- /dev/null +++ b/graphics/traveler/src/trv_rayrend.c @@ -0,0 +1,1337 @@ +/***************************************************************************** + * apps/graphics/traveler/src/trv_rayrend.c + * This file contains the functions needed to render a screen. + * + * Copyright (C) 2014 Gregory Nutt. All rights reserved. + * Author: Gregory Nutt + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in + * the documentation and/or other materials provided with the + * distribution. + * 3. Neither the name NuttX nor the names of its contributors may be + * used to endorse or promote products derived from this software + * without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, + * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, + * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS + * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED + * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN + * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + * + ****************************************************************************/ + +/**************************************************************************** + * Included Files + ****************************************************************************/ + +#include "trv_types.h" +#include "trv_debug.h" +#include "trv_world.h" +#include "trv_plane.h" +#include "trv_bitmaps.h" +#include "trv_graphics.h" +#include "trv_paltable.h" +#include "trv_trigtbl.h" +#include "trv_raycntl.h" +#include "trv_raycast.h" +#include "trv_rayrend.h" + +/**************************************************************************** + * Pre-processor Definitions + ****************************************************************************/ +/* These are sometimes useful during debug */ + +#define DISABLE_WALL_RENDING 0 +#define DISABLE_FLOOR_RENDING 0 + +/* The following macros perform division (using g_invsize[]) and then a + * rescaling by the approprate constants so that the texture index is + * byte aligned + */ + +#define TDIV(num,den,s) (((num) * g_invsize[den]) >> (s)) + +/* This macro just performs the division and leaves the result with eight + * (more) bits of fraction + */ + +#define DIV8(num,den) ((num) * g_invsize[den]) + +/* The following macro aligns a SMALL precision number to that the texture + * index is byte aligned + */ + +#define TALIGN(x,s) ((x) << (8-(s))) + +/**************************************************************************** + * Private Type Declarations + ****************************************************************************/ + +/* This union is used to manage texture indices */ + +union tex_ndx_u +{ + struct + { + uint8_t f; + uint8_t i; + } s; + int16_t w; +}; + +/**************************************************************************** + * Private Function Prototypes + ****************************************************************************/ + +static void trv_rend_zcell(uint8_t row, uint8_t col, uint8_t height, + uint8_t width); +static void trv_rend_zrow(uint8_t row, uint8_t col, uint8_t width); +static void trv_rend_zcol(uint8_t row, uint8_t col, uint8_t height); +static void trv_rend_zpixel(uint8_t row, uint8_t col); + +static void trv_rend_wall(uint8_t row, uint8_t col, uint8_t height, + uint8_t width); +static void trv_rend_wallrow(uint8_t row, uint8_t col, uint8_t width); +static void trv_rend_wallcol(uint8_t row, uint8_t col, uint8_t height); +static void trv_rend_wallpixel(uint8_t row, uint8_t col); + +/**************************************************************************** + * Private Data + ****************************************************************************/ + +/* The following array simply contains the inverted values of the integers + * from 0...VGULP_SIZE-1. The values in the table have 8 bits of fraction. + * The value for 0 is bogus! + */ + +static const uint8_t g_invsize[VGULP_SIZE] = +{ + 0xff, 0xff, 0x80, 0x55, 0x40, 0x33, 0x2b, 0x25, /* 0..7 */ + 0x20, 0x1c, 0x1a, 0x17, 0x15, 0x14, 0x12, 0x11 /* 8..15 */ +}; + +/**************************************************************************** + * Private Functions + ****************************************************************************/ + +/**************************************************************************** + * Name: trv_rend_zcell, trv_rend_zrow, trv_rend_zcol, trv_rend_zpixel + * + * Description: + * After matching hits have been obtained in all four corners of a cell, + * this function interpolates to complete the cell then transfers the cell + * to the double buffer. These are the general cases where no assumptions + * are made concerning the relative orientation of the cell and the texture. + * This general case is only used to texture floors and ceilings. + * + ****************************************************************************/ + +/* This version is for non-degenerate cell, i.e., height>1 and width>1 */ + +static void trv_rend_zcell(uint8_t row, uint8_t col, uint8_t height, uint8_t width) +{ +#if (!DISABLE_FLOOR_RENDING) + uint8_t i; + uint8_t j; + uint8_t endrow; + uint8_t endcol; + FAR uint8_t *palptr; + FAR uint8_t *outpixel; + uint8_t scale; + FAR trv_pixel_t *texture; + FAR struct trv_bitmap_s *bmp; + union tex_ndx_u xpos; + union tex_ndx_u ypos; + uint16_t tmask; + uint16_t tsize; + int16_t zone; + int16_t endzone; + int16_t zonestep; + int16_t hstart; + int16_t hrowstep; + int16_t hcolstep; + int16_t hstepincr; + int16_t vstart; + int16_t vrowstep; + int16_t vcolstep; + int16_t vstepincr; + int16_t tmpcolstep; + + /* Displace the double buffer pointer */ + + outpixel = &g_buffer_row[row][g_cell_column]; + + /* Point to the bitmap associated with the upper left pixel. Since + * all of the pixels in this cell are the same "hit," we don't have + * to recalculate this + */ + + if (IS_FRONT_HIT(&g_ray_hit[row][col])) + { + bmp = g_even_bitmaps[g_ray_hit[row][col].rect->texture]; + } + else + { + bmp = g_odd_bitmaps[g_ray_hit[row][col].rect->texture]; + } + + /* Get parameters associated with the size of the bitmap texture */ + + texture = bmp->bm; + tsize = bmp->log2h; + tmask = TMASK(tsize); + + /* Extract the texture scaling from the rectangle structure */ + + scale = g_ray_hit[row][col].rect->scale; + + /* Within this function, all references to height and width are really + * (height-1) and (width-1) + */ + + height--; + width--; + + /* Get the indices to the lower, right corner */ + + endcol = col + width; + endrow = row + height; + + /* Calculate the horizontal interpolation values */ + /* This is the H starting position (first row, first column) */ + + hstart = TALIGN(g_ray_hit[row][col].xpos, scale); + + /* This is the change in xpos per column in the first row */ + + hcolstep = + TDIV((g_ray_hit[row][endcol].xpos - g_ray_hit[row][col].xpos), + width, scale); + + /* This is the change in xpos per column in the last row */ + + tmpcolstep = + TDIV((g_ray_hit[endrow][endcol].xpos - g_ray_hit[endrow][col].xpos), + width, scale); + + /* This is the change in hcolstep per row */ + + hstepincr = (TDIV((tmpcolstep - hcolstep), height, scale) >> 8); + + /* This is the change in hstart for each row */ + + hrowstep = + TDIV((g_ray_hit[endrow][col].xpos - g_ray_hit[row][col].xpos), + height, scale); + + /* Calculate the vertical interpolation values */ + /* This is the V starting position (first row, first column) */ + + vstart = TALIGN(g_ray_hit[row][col].ypos, scale); + + /* This is the change in ypos per column in the first row */ + + vcolstep = + TDIV((g_ray_hit[row][endcol].ypos - g_ray_hit[row][col].ypos), + width, scale); + + /* This is the change in ypos per column in the last row */ + + tmpcolstep = + TDIV((g_ray_hit[endrow][endcol].ypos - g_ray_hit[endrow][col].ypos), + width, scale); + + /* This is the change in vcolstep per row */ + + vstepincr = (TDIV((tmpcolstep - vcolstep), height, scale) >> 8); + + /* This is the change in vstart for each row */ + + vrowstep = + TDIV((g_ray_hit[endrow][col].ypos - g_ray_hit[row][col].ypos), + height, scale); + + /* Determine the palette mapping table zone for each row */ + + if (IS_SHADED(g_ray_hit[row][col].rect)) + { + zone = GET_FZONE(g_ray_hit[row][col].xdist, g_ray_hit[row][col].ydist, 8); + endzone = GET_FZONE(g_ray_hit[endrow][col].xdist, g_ray_hit[endrow][col].ydist, 8); + zonestep = (DIV8((endzone - zone), height) >> 8); + } + else + { + zone = 0; + zonestep = 0; + } + + /* Now, interpolate to texture each row (vertical component) */ + + for (i = row; i <= endrow; i++) + { + /* Set the initial horizontal & vertical offset */ + + xpos.w = hstart; + ypos.w = vstart; + + /* Get the palette map to use on this row */ + + palptr = GET_PALPTR((zone >> 8)); + + /* Interpolate to texture each column in the row */ + + for (j = col; j <= endcol; j++) + { + /* Transfer the pixel at this interpolated position */ + + outpixel[j] = palptr[texture[TNDX(xpos.s.i, ypos.s.i, tsize, tmask)]]; + + /* Now Calculate the horizontal position for the next step */ + + xpos.w += hcolstep; + ypos.w += vcolstep; + } + + /* Point to the next row */ + + outpixel += TRV_SCREEN_WIDTH; + + /* Calculate the vertical position for the next step */ + + hstart += hrowstep; + vstart += vrowstep; + + /* Adjust the step sizes to use on the next row */ + + hcolstep += hstepincr; + vcolstep += vstepincr; + + /* Get the zone to use on the next row */ + + zone += zonestep; + } +#endif +} + +/* This version is for horizontal lines, i.e., height==1 and width>1 */ + +static void trv_rend_zrow(uint8_t row, uint8_t col, uint8_t width) +{ +#if (!DISABLE_FLOOR_RENDING) + uint8_t j; + uint8_t endcol; + FAR uint8_t *palptr; + FAR uint8_t *outpixel; + uint8_t scale; + FAR trv_pixel_t *texture; + FAR struct trv_bitmap_s *bmp; + union tex_ndx_u xpos; + union tex_ndx_u ypos; + uint16_t tmask; + uint16_t tsize; + int16_t zone; + int16_t hcolstep; + int16_t vcolstep; + + /* Displace the double buffer pointer */ + + outpixel = &g_buffer_row[row][g_cell_column]; + + /* Point to the bitmap associated with the left pixel. Since + * all of the pixels in this row are the same "hit," we don't have + * to recalculate this + */ + + if (IS_FRONT_HIT(&g_ray_hit[row][col])) + { + bmp = g_even_bitmaps[g_ray_hit[row][col].rect->texture]; + } + else + { + bmp = g_odd_bitmaps[g_ray_hit[row][col].rect->texture]; + } + + /* Get parameters associated with the size of the bitmap texture */ + + texture = bmp->bm; + tsize = bmp->log2h; + tmask = TMASK(tsize); + + /* Extract the texture scaling from the rectangle structure */ + + scale = g_ray_hit[row][col].rect->scale; + + /* Get the a pointer to the palette mapping table */ + + if (IS_SHADED(g_ray_hit[row][col].rect)) + { + zone = GET_ZONE(g_ray_hit[row][col].xdist, g_ray_hit[row][col].ydist); + palptr = GET_PALPTR(zone); + } + else + { + palptr = GET_PALPTR(0); + } + + /* Within this function, all references to width are really + * (width-1) + */ + + width--; + + /* Get the index to the right side */ + + endcol = col + width; + + /* Calculate the horizontal interpolation values */ + /* This is the H starting position (first column) */ + + xpos.w = TALIGN(g_ray_hit[row][col].xpos, scale); + + /* This is the change in xpos per column */ + + hcolstep = + TDIV((g_ray_hit[row][endcol].xpos - g_ray_hit[row][col].xpos), + width, scale); + + /* Calculate the vertical interpolation values */ + /* This is the V starting position (first column) */ + + ypos.w = TALIGN(g_ray_hit[row][col].ypos, scale); + + /* This is the change in ypos per column */ + + vcolstep = + TDIV((g_ray_hit[row][endcol].ypos - g_ray_hit[row][col].ypos), + width, scale); + + /* Interpolate to texture each column in the row */ + + for (j = col; j <= endcol; j++) + { + /* Transfer the pixel at this interpolated position */ + + outpixel[j] = palptr[texture[TNDX(xpos.s.i, ypos.s.i, tsize, tmask)]]; + + /* Now Calculate the horizontal position for the next step */ + + xpos.w += hcolstep; + ypos.w += vcolstep; + } +#endif +} + +/* This version is for vertical lines, i.e., height>1 and width==1 */ + +static void trv_rend_zcol(uint8_t row, uint8_t col, uint8_t height) +{ +#if (!DISABLE_FLOOR_RENDING) + uint8_t i, endrow; + FAR uint8_t *palptr; + FAR uint8_t *outpixel; + uint8_t scale; + FAR trv_pixel_t *texture; + FAR struct trv_bitmap_s *bmp; + union tex_ndx_u xpos; + union tex_ndx_u ypos; + uint16_t tmask; + uint16_t tsize; + int16_t zone; + int16_t hrowstep; + int16_t vrowstep; + + /* Displace the double buffer pointer */ + + outpixel = &g_buffer_row[row][g_cell_column+col]; + + /* Point to the bitmap associated with the upper pixel. Since + * all of the pixels in this column are the same "hit," we don't have + * to recalculate this + */ + + if (IS_FRONT_HIT(&g_ray_hit[row][col])) + { + bmp = g_even_bitmaps[g_ray_hit[row][col].rect->texture]; + } + else + { + bmp = g_odd_bitmaps[g_ray_hit[row][col].rect->texture]; + } + + /* Get parameters associated with the size of the bitmap texture */ + + texture = bmp->bm; + tsize = bmp->log2h; + tmask = TMASK(tsize); + + /* Extract the texture scaling from the rectangle structure */ + + scale = g_ray_hit[row][col].rect->scale; + + /* Get the a pointer to the palette mapping table */ + + if (IS_SHADED(g_ray_hit[row][col].rect)) + { + zone = GET_ZONE(g_ray_hit[row][col].xdist, g_ray_hit[row][col].ydist); + palptr = GET_PALPTR(zone); + } + else + { + palptr = GET_PALPTR(0); + } + + /* Within this function, all references to height are really + * (height-1) + */ + + height--; + + /* Get the indices to the lower end */ + + endrow = row + height; + + /* Calculate the horizontal interpolation values */ + /* This is the H starting position (first row) */ + + xpos.w = TALIGN(g_ray_hit[row][col].xpos, scale); + + /* This is the change in xpos for each row */ + + hrowstep = + TDIV((g_ray_hit[endrow][col].xpos - g_ray_hit[row][col].xpos), + height, scale); + + /* Calculate the vertical interpolation values */ + /* This is the V starting position (first row) */ + + ypos.w = TALIGN(g_ray_hit[row][col].ypos, scale); + + /* This is the change in ypos for each row */ + + vrowstep = + TDIV((g_ray_hit[endrow][col].ypos - g_ray_hit[row][col].ypos), + height, scale); + + /* Now, interpolate to texture each row (vertical component) */ + + for (i = row; i <= endrow; i++) + { + /* Transfer the pixel at this interpolated position */ + + *outpixel = palptr[texture[TNDX(xpos.s.i, ypos.s.i, tsize, tmask)]]; + + /* Point to the next row */ + + outpixel += TRV_SCREEN_WIDTH; + + /* Calculate the vertical position for the next step */ + + xpos.w += hrowstep; + ypos.w += vrowstep; + } +#endif +} + +/* This version is for a single pixel, i.e., height==1 and width==1 */ + +static void trv_rend_zpixel(uint8_t row, uint8_t col) +{ +#if (!DISABLE_FLOOR_RENDING) + FAR uint8_t *palptr; + FAR trv_pixel_t *texture; + FAR struct trv_bitmap_s *bmp; + uint16_t tmask; + uint16_t tsize; + int16_t zone; + + /* Get the a pointer to the palette mapping table */ + + if (IS_SHADED(g_ray_hit[row][col].rect)) + { + zone = GET_ZONE(g_ray_hit[row][col].xdist, g_ray_hit[row][col].ydist); + palptr = GET_PALPTR(zone); + } + else + { + palptr = GET_PALPTR(0); + } + + /* Point to the bitmap associated with the upper left pixel. */ + + if (IS_FRONT_HIT(&g_ray_hit[row][col])) + { + bmp = g_even_bitmaps[g_ray_hit[row][col].rect->texture]; + } + else + { + bmp = g_odd_bitmaps[g_ray_hit[row][col].rect->texture]; + } + + /* Get parameters associated with the size of the bitmap texture */ + + texture = bmp->bm; + tsize = bmp->log2h; + tmask = TMASK(tsize); + + g_buffer_row[row][g_cell_column+col] = + palptr[texture[TNDX(g_ray_hit[row][col].xpos, g_ray_hit[row][col].ypos, + tsize, tmask)]]; +#endif +} + +/**************************************************************************** + * Name: trv_rend_wall, trv_rend_wallrow, trv_rend_wallcol, diplayWallPixel + * + * Description: + * After matching hits have been obtained in all four corners of a cell, + * this function interpolates to complete the cell then transfers the cell + * to the double buffer. These special simplifications for use on on + * vertical (X or Y) walls. In this case, we can assume that: + * + * g_ray_hit[row][col].xpos == g_ray_hit[row+height-1][col] + * g_ray_hit[row][col+width-1].xpos == g_ray_hit[row+height-1][col+width-1] + * + * In addition to these simplifications, these functions include the + * added complications of handling internal INVISIBLE_PIXELs which may + * occur within TRANSPARENT_WALLs. + * + ****************************************************************************/ + +/* This version is for non-degenerate cell, i.e., height>1 and width>1 */ + +static void trv_rend_wall(uint8_t row, uint8_t col, + uint8_t height, uint8_t width) +{ +#if (!DISABLE_WALL_RENDING) + uint8_t i, j; + uint8_t endrow; + uint8_t endcol; + uint8_t inpixel; + FAR uint8_t *palptr; + FAR uint8_t *outpixel; + uint8_t scale; + FAR trv_pixel_t *texture; + FAR struct trv_bitmap_s *bmp; + union tex_ndx_u xpos; + union tex_ndx_u ypos; + uint16_t tmask; + uint16_t tsize; + int16_t zone; + int16_t hstart; + int16_t hcolstep; + int16_t vstart; + int16_t vrowstep; + int16_t vcolstep; + int16_t vstepincr; + int16_t tmpcolstep; + + /* Displace the double buffer pointer */ + + outpixel = &g_buffer_row[row][g_cell_column]; + + /* Point to the bitmap associated with the upper left pixel. Since + * all of the pixels in this cell are the same "hit," we don't have + * to recalculate this + */ + + if (IS_FRONT_HIT(&g_ray_hit[row][col])) + { + bmp = g_even_bitmaps[g_ray_hit[row][col].rect->texture]; + } + else + { + bmp = g_odd_bitmaps[g_ray_hit[row][col].rect->texture]; + } + + /* Get parameters associated with the size of the bitmap texture */ + + texture = bmp->bm; + tsize = bmp->log2h; + tmask = TMASK(tsize); + + /* Extract the texture scaling from the rectangle structure */ + + scale = g_ray_hit[row][col].rect->scale; + + /* Get the a pointer to the palette mapping table */ + + if (IS_SHADED(g_ray_hit[row][col].rect)) + { + zone = GET_ZONE(g_ray_hit[row][col].xdist, g_ray_hit[row][col].ydist); + palptr = GET_PALPTR(zone); + } + else + { + palptr = GET_PALPTR(0); + } + + /* Within this function, all references to height and width are really + * (height-1) and (width-1) + */ + + height--; + width--; + + /* Get the indices to the lower, right corner */ + + endcol = col + width; + endrow = row + height; + + /* Calculate the horizontal interpolation values */ + /* This is the H starting position (first row, first column) */ + + hstart = TALIGN(g_ray_hit[row][col].xpos, scale); + + /* This is the change in xpos per column in the first row */ + + hcolstep = + TDIV((g_ray_hit[row][endcol].xpos - g_ray_hit[row][col].xpos), + width, scale); + + /* Calculate the vertical interpolation values */ + /* This is the V starting position (first row, first column) */ + + vstart = TALIGN(g_ray_hit[row][col].ypos, scale); + + /* This is the change in ypos per column in the first row */ + + vcolstep = + TDIV((g_ray_hit[row][endcol].ypos - g_ray_hit[row][col].ypos), + width, scale); + + /* This is the change in ypos per column in the last row */ + + tmpcolstep = + TDIV((g_ray_hit[endrow][endcol].ypos - g_ray_hit[endrow][col].ypos), + width, scale); + + /* This is the change in vcolstep per row */ + + vstepincr = (TDIV((tmpcolstep - vcolstep), height, scale) >> 8); + + /* This is the change in vstart for each row */ + + vrowstep = + TDIV((g_ray_hit[endrow][col].ypos - g_ray_hit[row][col].ypos), + height, scale); + + /* Now, interpolate to texture each row (vertical component) */ + + for (i = row; i <= endrow; i++) + { + /* Set the initial horizontal & vertical offset */ + + xpos.w = hstart; + ypos.w = vstart; + + /* Interpolate to texture each column in the row */ + + for (j = col; j <= endcol; j++) + { + /* Extract the pixel from the texture */ + + inpixel = texture[TNDX(xpos.s.i, ypos.s.i, tsize, tmask)]; + + /* If this is an INVISIBLE_PIXEL in a TRANSPARENT_WALL, then + * we will have to take some pretty extreme measures to get the + * correct value of the pixel + */ + + if ((inpixel == INVISIBLE_PIXEL) && + (IS_TRANSPARENT(g_ray_hit[row][col].rect))) + { + /* Check if we hit anything */ + + if ((inpixel = trv_get_texture(i, j)) != INVISIBLE_PIXEL) + { + /* Map the normal pixel and transfer the pixel at this + * interpolated position + */ + + outpixel[j] = inpixel; + } + } + else + { + /* Map the normal pixel and transfer the pixel at this + * interpolated position + */ + + outpixel[j] = palptr[inpixel]; + } + + /* Now Calculate the horizontal position for the next step */ + + xpos.w += hcolstep; + ypos.w += vcolstep; + } + + /* Point to the next row */ + + outpixel += TRV_SCREEN_WIDTH; + + /* Calculate the vertical position for the next step */ + + vstart += vrowstep; + + /* Adjust the step sizes to use on the next row */ + + vcolstep += vstepincr; + } +#endif +} + +/* This version is for horizontal lines, i.e., height==1 and width>1 */ + +static void trv_rend_wallrow(uint8_t row, uint8_t col, uint8_t width) +{ +#if (!DISABLE_WALL_RENDING) + uint8_t j; + uint8_t endcol; + uint8_t inpixel; + FAR uint8_t *palptr; + FAR uint8_t *outpixel; + uint8_t scale; + FAR trv_pixel_t *texture; + FAR struct trv_bitmap_s *bmp; + union tex_ndx_u xpos; + union tex_ndx_u ypos; + uint16_t tmask; + uint16_t tsize; + int16_t zone; + int16_t hcolstep; + int16_t vcolstep; + + /* Displace the double buffer pointer */ + + outpixel = &g_buffer_row[row][g_cell_column]; + + /* Point to the bitmap associated with the left pixel. Since + * all of the pixels in this row are the same "hit," we don't have + * to recalculate this + */ + + if (IS_FRONT_HIT(&g_ray_hit[row][col])) + { + bmp = g_even_bitmaps[g_ray_hit[row][col].rect->texture]; + } + else + { + bmp = g_odd_bitmaps[g_ray_hit[row][col].rect->texture]; + } + + /* Get parameters associated with the size of the bitmap texture */ + + texture = bmp->bm; + tsize = bmp->log2h; + tmask = TMASK(tsize); + + /* Extract the texture scaling from the rectangle structure */ + + scale = g_ray_hit[row][col].rect->scale; + + /* Get the a pointer to the palette mapping table */ + + if (IS_SHADED(g_ray_hit[row][col].rect)) + { + zone = GET_ZONE(g_ray_hit[row][col].xdist, g_ray_hit[row][col].ydist); + palptr = GET_PALPTR(zone); + } + else + { + palptr = GET_PALPTR(0); + } + + /* Within this function, all references to width are really + * (width-1) + */ + + width--; + + /* Get the index to the right side */ + + endcol = col + width; + + /* Calculate the horizontal interpolation values */ + /* This is the H starting position (first column) */ + + xpos.w = TALIGN(g_ray_hit[row][col].xpos, scale); + + /* This is the change in xpos per column */ + + hcolstep = + TDIV((g_ray_hit[row][endcol].xpos - g_ray_hit[row][col].xpos), + width, scale); + + /* Calculate the vertical interpolation values */ + /* This is the V starting position (first column) */ + + ypos.w = TALIGN(g_ray_hit[row][col].ypos, scale); + + /* This is the change in ypos per column */ + + vcolstep = + TDIV((g_ray_hit[row][endcol].ypos - g_ray_hit[row][col].ypos), + width, scale); + + /* Interpolate to texture each column in the row */ + + for (j = col; j <= endcol; j++) + { + /* Extract the pixel from the texture */ + + inpixel = texture[TNDX(xpos.s.i, ypos.s.i, tsize, tmask)]; + + /* If this is an INVISIBLE_PIXEL in a TRANSPARENT_WALL, then + * we will have to take some pretty extreme measures to get the + * correct value of the pixel + */ + + if ((inpixel == INVISIBLE_PIXEL) && + (IS_TRANSPARENT(g_ray_hit[row][col].rect))) + { + /* Cast another ray and see if we hit anything */ + + if ((inpixel = trv_get_texture(row, j)) != INVISIBLE_PIXEL) + { + /* Map the normal pixel and transfer the pixel at this + * interpolated position + */ + + outpixel[j] = inpixel; + } + } + else + { + /* Map the normal pixel and transfer the pixel at this + * interpolated position + */ + + outpixel[j] = palptr[inpixel]; + } + + /* Now Calculate the horizontal position for the next step */ + + xpos.w += hcolstep; + ypos.w += vcolstep; + } +#endif +} + +/* This version is for vertical line, i.e., height>1 and width==1 */ + +static void trv_rend_wallcol(uint8_t row, uint8_t col, uint8_t height) +{ +#if (!DISABLE_WALL_RENDING) + uint8_t i; + uint8_t endrow; + uint8_t inpixel; + FAR uint8_t *palptr; + FAR uint8_t *outpixel; + uint8_t scale; + FAR trv_pixel_t *texture; + FAR struct trv_bitmap_s *bmp; + union tex_ndx_u ypos; + uint16_t tmask; + uint16_t tsize; + int16_t zone; + int16_t xpos; + int16_t vrowstep; + + /* Displace the double buffer pointer */ + + outpixel = &g_buffer_row[row][g_cell_column+col]; + + /* Point to the bitmap associated with the upper pixel. Since + * all of the pixels in this cell are the same "hit," we don't have + * to recalculate this + */ + + if (IS_FRONT_HIT(&g_ray_hit[row][col])) + { + bmp = g_even_bitmaps[g_ray_hit[row][col].rect->texture]; + } + else + { + bmp = g_odd_bitmaps[g_ray_hit[row][col].rect->texture]; + } + + /* Get parameters associated with the size of the bitmap texture */ + + texture = bmp->bm; + tsize = bmp->log2h; + tmask = TMASK(tsize); + + /* Extract the texture scaling from the rectangle structure */ + + scale = g_ray_hit[row][col].rect->scale; + + /* Get the a pointer to the palette mapping table */ + + if (IS_SHADED(g_ray_hit[row][col].rect)) + { + zone = GET_ZONE(g_ray_hit[row][col].xdist, g_ray_hit[row][col].ydist); + palptr = GET_PALPTR(zone); + } + else + { + palptr = GET_PALPTR(0); + } + + /* Within this function, all references to height are really + * (height-1) + */ + + height--; + + /* Get the indices to the lower end */ + + endrow = row + height; + + /* Calculate the horizontal interpolation values */ + + xpos = sFRAC(g_ray_hit[row][col].xpos >> scale); + + /* Calculate the vertical interpolation values */ + /* This is the V starting position (first row, first column) */ + + ypos.w = TALIGN(g_ray_hit[row][col].ypos, scale); + + /* This is the change in ypos for each row */ + + vrowstep = + TDIV((g_ray_hit[endrow][col].ypos - g_ray_hit[row][col].ypos), + height, scale); + + /* Now, interpolate to texture the vertical line */ + + for (i = row; i <= endrow; i++) + { + /* Extract the pixel from the texture */ + + inpixel = texture[TNDX(xpos, ypos.s.i, tsize, tmask)]; + + /* If this is an INVISIBLE_PIXEL in a TRANSPARENT_WALL, then + * we will have to take some pretty extreme measures to get the + * correct value of the pixel + */ + + if ((inpixel == INVISIBLE_PIXEL) && + (IS_TRANSPARENT(g_ray_hit[row][col].rect))) + { + /* Check if we hit anything */ + + if ((inpixel = trv_get_texture(i, col)) != INVISIBLE_PIXEL) + { + /* Map the normal pixel and transfer the pixel at this + * interpolated position + */ + + *outpixel = inpixel; + } + } + else + { + /* Map the normal pixel and transfer the pixel at this + * interpolated position + */ + + *outpixel = palptr[inpixel]; + } + + /* Point to the next row */ + + outpixel += TRV_SCREEN_WIDTH; + + /* Calculate the vertical position for the next step */ + + ypos.w += vrowstep; + } +#endif +} + +/* This version is for a single pixel, i.e., height==1 and width==1 */ + +static void trv_rend_wallpixel(uint8_t row, uint8_t col) +{ +#if (!DISABLE_WALL_RENDING) + uint8_t *palptr; + int16_t zone; + + /* Get the a pointer to the palette mapping table */ + + if (IS_SHADED(g_ray_hit[row][col].rect)) + { + zone = GET_ZONE(g_ray_hit[row][col].xdist, g_ray_hit[row][col].ydist); + palptr = GET_PALPTR(zone); + } + else + { + palptr = GET_PALPTR(0); + } + + /* The map and transfer the pixel to the display buffer */ + + if (IS_FRONT_HIT(&g_ray_hit[row][col])) + { + g_buffer_row[row][g_cell_column+col] = + palptr[GET_FRONT_PIXEL(g_ray_hit[row][col].rect, + g_ray_hit[row][col].xpos, + g_ray_hit[row][col].ypos)]; + } + else + { + g_buffer_row[row][g_cell_column+col] = + palptr[GET_BACK_PIXEL(g_ray_hit[row][col].rect, + g_ray_hit[row][col].xpos, + g_ray_hit[row][col].ypos)]; + } +#endif +} + +/**************************************************************************** + * Public Functions + ****************************************************************************/ + +/**************************************************************************** + * Name: trv_rend_backdrop + * Description: + * + * Clear the screen and draw the sky and ground using 32 bit transfers. + * + ****************************************************************************/ + +void trv_rend_backdrop(FAR struct trv_camera_s *camera, + FAR struct trv_graphics_info_s *ginfo) +{ +#ifdef ENABLE_VIDEO + FAR uint32_t *dest; + uint32_t plotpixels; + int32_t skyrows; + uint16_t buffersize; + uint16_t n; + int16_t pitchoffset; + + /* The destination of the transfer is the screen buffer */ + + dest = (uint32_t*)ginfo->swbuffer; + + /* Convert the pitch angle into signed screen offset */ + + if ((pitchoffset = camera->pitch) > ANGLE_90) + { + pitchoffset -= ANGLE_360; + } + + /* Determine the size of the "sky" buffer in rows. Positive pitchoffset + * means we are looking up. In this case the sky gets bigger. + */ + + skyrows = WINDOW_MIDDLE + pitchoffset; + + /* Handle the case where we are looking down and do not see any of the + * sky + */ + + if (skyrows <= 0) + { + /* No sky rows -- No sky buffersize */ + + skyrows = buffersize = 0; + } + + /* Copy the sky color into the top half of the screen */ + + else + { + /* Handle the case where we are looking up and see only the sky */ + + if (skyrows > WINDOW_HEIGHT) + { + skyrows = WINDOW_HEIGHT; + } + + /* Determine the size of the "sky" buffer in 32-bit units */ + + buffersize = (TRV_SCREEN_WIDTH/4) * skyrows; + + /* Combine the sky color into 32-bit "quadruple pixels" */ + + plotpixels = (((uint32_t)g_sky_color << 24) | + ((uint32_t)g_sky_color << 16) | + ((uint32_t)g_sky_color << 8) | + (uint32_t)g_sky_color); + + /* Then transfer the "sky" */ + + for (n = 0; n < buffersize; n++) + { + *dest++ = plotpixels; + } + } + + /* Copy the ground color into the bottom half of the screen */ + + if (skyrows < WINDOW_HEIGHT) + { + /* Determine the size of the "ground" buffer in 32-bit units */ + + buffersize = (TRV_SCREEN_WIDTH/4) * (WINDOW_HEIGHT - skyrows); + + /* Combine the ground color into 32-bit "quadruple pixels" */ + + plotpixels = (((uint32_t)g_ground_color << 24) | + ((uint32_t)g_ground_color << 16) | + ((uint32_t)g_ground_color << 8) | + (uint32_t)g_ground_color); + + /* Then transfer the "ground" */ + + for (n = 0; n < buffersize; n++) + { + *dest++ = plotpixels; + } + } +#endif +} + +/**************************************************************************** + * Name: trv_rend_cell, trv_rend_row, trv_rend_column, trv_rend_pixel + * + * Description: + * After matching hits have been obtained in all four corners of a cell, + * this function interpolates to complete the cell then transfers the cell + * to the double buffer. + * + ****************************************************************************/ + +/* This version is for non-degenerate cell, i.e., height>1 and width>1 */ + +void trv_rend_cell(uint8_t row, uint8_t col, uint8_t height, uint8_t width) +{ + /* If the cell is visible, then put it in the off-screen buffer. + * Otherwise, just drop it on the floor + */ + + if (g_ray_hit[row][col].rect) + { + /* Apply texturing... special case for hits on floor or ceiling */ + + if (IS_ZRAY_HIT(&g_ray_hit[row][col])) + { + trv_rend_zcell(row, col, height, width); + } + else + { + trv_rend_wall(row, col, height, width); + } + } +} + +/* This version is for horizontal lines, i.e., height==1 and width>1 */ + +void trv_rend_row(uint8_t row, uint8_t col, uint8_t width) +{ + /* If the cell is visible, then put it in the off-screen buffer. + * Otherwise, just drop it on the floor + */ + + if (g_ray_hit[row][col].rect) + { + /* Apply texturing... special case for hits on floor or ceiling */ + + if (IS_ZRAY_HIT(&g_ray_hit[row][col])) + { + trv_rend_zrow(row, col, width); + } + else + { + trv_rend_wallrow(row, col, width); + } + } +} + +/* This version is for vertical lines, i.e., height>1 and width==1 */ + +void trv_rend_column(uint8_t row, uint8_t col, uint8_t height) +{ + /* If the cell is visible, then put it in the off-screen buffer. + * Otherwise, just drop it on the floor + */ + + if (g_ray_hit[row][col].rect) + { + /* Apply texturing... special case for hits on floor or ceiling */ + + if (IS_ZRAY_HIT(&g_ray_hit[row][col])) + { + trv_rend_zcol(row, col, height); + } + else + { + trv_rend_wallcol(row, col, height); + } + } +} + +/* This version is for a single pixel, i.e., height==1 and width==1 */ + +void trv_rend_pixel(uint8_t row, uint8_t col) +{ + /* If the cell is visible, then put it in the off-screen buffer. + * Otherwise, just drop it on the floor + */ + + if (g_ray_hit[row][col].rect) + { + /* Apply texturing... special case for hits on floor or ceiling */ + + if (IS_ZRAY_HIT(&g_ray_hit[row][col])) + { + trv_rend_zpixel(row, col); + } + else + { + trv_rend_wallpixel(row, col); + } + } +} + +/**************************************************************************** + * Name: trv_get_rectpixel + * + * Description: + * Returns the pixel for a hit at (xpos, ypos) on rect. + * + ****************************************************************************/ + +trv_pixel_t trv_get_rectpixel(int16_t xpos, int16_t ypos, + FAR struct trv_bitmap_s *bmp, uint8_t scale) +{ + uint16_t tmask; + uint16_t tsize; + + /* Get parameters associated with the size of the bitmap texture */ + + tsize = bmp->log2h; + tmask = TMASK(tsize); + + /* Return the texture code at this position */ + + return(bmp->bm[TNDX((xpos >> scale), (ypos >> scale), tsize, tmask)]); +} + diff --git a/graphics/traveler/src/trv_texturefile.c b/graphics/traveler/src/trv_texturefile.c new file mode 100644 index 000000000..8fb5d2404 --- /dev/null +++ b/graphics/traveler/src/trv_texturefile.c @@ -0,0 +1,201 @@ +/******************************************************************************* + * apps/graphics/traveler/src/trv_texturefile.c + * + * Copyright (C) 2014 Gregory Nutt. All rights reserved. + * Author: Gregory Nutt + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in + * the documentation and/or other materials provided with the + * distribution. + * 3. Neither the name NuttX nor the names of its contributors may be + * used to endorse or promote products derived from this software + * without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, + * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, + * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS + * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED + * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN + * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + * + ****************************************************************************/ + +/**************************************************************************** + * Included files + ****************************************************************************/ + +#include "trv_types.h" +#include "trv_bitmaps.h" +#include "trv_mem.h" +#include "trv_color.h" +#include "trv_graphicfile.h" +#include "trv_fsutils.h" + +#include +#include + +/**************************************************************************** + * Pre-processor Definitions + ****************************************************************************/ + +#define COLOR_LEVELS 6 + +#undef MIN +#undef MAX +#define MIN(a,b) ((a) < (b) ? (a) : (b)) +#define MAX(a,b) ((a) > (b) ? (a) : (b)) + +/**************************************************************************** + * Private Functions + ****************************************************************************/ + +/**************************************************************************** + * Name: trv_log2 + * + * Description: + * Return the log base 2 of the argument, or -1 if the argument is not + * an integer power of 2. + * + ***************************************************************************/ + +static int trv_log2(uint16_t x) +{ + int i; + + if (x != 0) + { + for (i = 0; i < 16 && (x & 1) == 0; i++, x >>= 1); + } + + return (x == 1) ? i : -EINVAL; +} + +/**************************************************************************** + * Name: trv_new_texture + * + * Description: + ***************************************************************************/ + +static FAR struct trv_bitmap_s * +trv_new_texture(uint16_t width, uint16_t height) +{ + struct trv_bitmap_s *bitmap; + int log2h; + int log2w; + + /* The height and width should be powers of two for efficient texture + * mapping. Here, we enforce this. + */ + + log2w = trv_log2(width); + if (log2w < 0) + { + fprintf(stderr, "ERROR: Width is not a power of 2: %u\n", width); + return NULL; + } + + log2h = trv_log2(height); + if (log2h < 0) + { + fprintf(stderr, "ERROR: Height is not a power of 2: %u\n", height); + return NULL; + } + + /* Allocate the bitmap container */ + + bitmap = (FAR struct trv_bitmap_s*)trv_malloc(sizeof(struct trv_bitmap_s)); + if (!bitmap) + { + fprintf(stderr, "ERROR: Failed to allocate bitmap structure\n", + height, width); + return NULL; + } + + /* Allocate the bitmap texture memory */ + + bitmap->bm = (trv_pixel_t*)trv_malloc(height * width * sizeof(trv_pixel_t)); + if (!bitmap->bm) + { + fprintf(stderr, "ERROR: Failed to allocate bitmap: %u x %u x %u\n", + height, width, sizeof(trv_pixel_t)); + trv_free(bitmap); + return NULL; + } + + /* Initialize the bitmap container */ + + bitmap->w = width; + bitmap->h = height; + bitmap->log2h = log2h; + + return bitmap; +} + +/**************************************************************************** + * Name: trv_quantize_texture + * Description: + ***************************************************************************/ + +static void trv_quantize_texture(FAR struct trv_graphicfile_s *gfile, + FAR struct trv_bitmap_s *bitmap) +{ + FAR trv_pixel_t *destpixel = bitmap->bm; + struct trv_color_rgb_s pixel; + int x; + int y; + + for (x = gfile->width - 1; x >= 0; x--) + { + for (y = gfile->height - 1; y >= 0; y--) + { + pixel = trv_graphicfile_pixel(gfile, x, y); + *destpixel++ = trv_color_rgb2pixel(&pixel); + } + } +} + +/**************************************************************************** + * Public Functions + ****************************************************************************/ + +/**************************************************************************** + * Name: trv_read_texture + * + * Description: + * + ***************************************************************************/ + +FAR struct trv_bitmap_s *trv_read_texture(char *filename) +{ + FAR struct trv_graphicfile_s *gfile; + FAR struct trv_bitmap_s *bitmap; + + gfile = tvr_graphicfile_read(filename); + if (gfile == NULL) + { + fprintf(stderr, "Error reading texture %s.", filename); + return NULL; + } + + bitmap = trv_new_texture(gfile->width, gfile->height); + if (bitmap) + { + trv_quantize_texture(gfile, bitmap); + } + + trv_graphicfile_free(gfile); + return bitmap; +} diff --git a/graphics/traveler/src/trv_trigtbl.c b/graphics/traveler/src/trv_trigtbl.c new file mode 100644 index 000000000..5e58ef85c --- /dev/null +++ b/graphics/traveler/src/trv_trigtbl.c @@ -0,0 +1,1287 @@ +/**************************************************************************** + * apps/graphics/traveler/src/trv_trigtbl.c + * Provides look-up tables for trigonometric functions. + * + * Copyright (C) 2014 Gregory Nutt. All rights reserved. + * Author: Gregory Nutt + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in + * the documentation and/or other materials provided with the + * distribution. + * 3. Neither the name NuttX nor the names of its contributors may be + * used to endorse or promote products derived from this software + * without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, + * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, + * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS + * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED + * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN + * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + * + ****************************************************************************/ + +/**************************************************************************** + * Included Files + ****************************************************************************/ + +#include "trv_types.h" +#include "trv_trigtbl.h" + +/**************************************************************************** + * Public Data + ****************************************************************************/ + +/* This is TAN lookup table. NOTES: + * 1. The index ranges in value from 0 to 3*PI/2-1. + * 2. Values in the range PI to 2*PI can be obtained from TAN(x) = TAN(x-PI). + * 3. The values obtain from the table have 12 bits of fraction. + * 4. COT(x) = 1/TAN(x) = TAN(90-X) + * 5. TAN(PI/2) is not defined (zero substituted). */ + +const int32_t g_tan_table[PI + HALFPI + 1] = +{ + 0x00000000, 0x0000000d, 0x0000001b, 0x00000028, 0x00000036, + 0x00000043, 0x00000050, 0x0000005e, 0x0000006b, 0x00000079, + 0x00000086, 0x00000094, 0x000000a1, 0x000000ae, 0x000000bc, + 0x000000c9, 0x000000d7, 0x000000e4, 0x000000f2, 0x000000ff, + 0x0000010c, 0x0000011a, 0x00000127, 0x00000135, 0x00000142, + 0x00000150, 0x0000015d, 0x0000016b, 0x00000178, 0x00000186, + 0x00000193, 0x000001a1, 0x000001af, 0x000001bc, 0x000001ca, + 0x000001d7, 0x000001e5, 0x000001f2, 0x00000200, 0x0000020e, + + 0x0000021b, 0x00000229, 0x00000237, 0x00000244, 0x00000252, + 0x00000260, 0x0000026d, 0x0000027b, 0x00000289, 0x00000296, + 0x000002a4, 0x000002b2, 0x000002c0, 0x000002ce, 0x000002db, + 0x000002e9, 0x000002f7, 0x00000305, 0x00000313, 0x00000321, + 0x0000032f, 0x0000033d, 0x0000034b, 0x00000359, 0x00000367, + 0x00000375, 0x00000383, 0x00000391, 0x0000039f, 0x000003ad, + 0x000003bb, 0x000003c9, 0x000003d7, 0x000003e6, 0x000003f4, + 0x00000402, 0x00000410, 0x0000041f, 0x0000042d, 0x0000043b, + + 0x0000044a, 0x00000458, 0x00000466, 0x00000475, 0x00000483, + 0x00000492, 0x000004a0, 0x000004af, 0x000004bd, 0x000004cc, + 0x000004db, 0x000004e9, 0x000004f8, 0x00000507, 0x00000515, + 0x00000524, 0x00000533, 0x00000542, 0x00000551, 0x0000055f, + 0x0000056e, 0x0000057d, 0x0000058c, 0x0000059b, 0x000005aa, + 0x000005ba, 0x000005c9, 0x000005d8, 0x000005e7, 0x000005f6, + 0x00000606, 0x00000615, 0x00000624, 0x00000634, 0x00000643, + 0x00000653, 0x00000662, 0x00000672, 0x00000681, 0x00000691, + + 0x000006a1, 0x000006b0, 0x000006c0, 0x000006d0, 0x000006e0, + 0x000006f0, 0x00000700, 0x00000710, 0x00000720, 0x00000730, + 0x00000740, 0x00000750, 0x00000760, 0x00000771, 0x00000781, + 0x00000791, 0x000007a2, 0x000007b2, 0x000007c3, 0x000007d3, + 0x000007e4, 0x000007f5, 0x00000805, 0x00000816, 0x00000827, + 0x00000838, 0x00000849, 0x0000085a, 0x0000086b, 0x0000087c, + 0x0000088d, 0x0000089f, 0x000008b0, 0x000008c1, 0x000008d3, + 0x000008e4, 0x000008f6, 0x00000908, 0x00000919, 0x0000092b, + + 0x0000093d, 0x0000094f, 0x00000961, 0x00000973, 0x00000985, + 0x00000997, 0x000009a9, 0x000009bc, 0x000009ce, 0x000009e1, + 0x000009f3, 0x00000a06, 0x00000a18, 0x00000a2b, 0x00000a3e, + 0x00000a51, 0x00000a64, 0x00000a77, 0x00000a8a, 0x00000a9e, + 0x00000ab1, 0x00000ac4, 0x00000ad8, 0x00000aeb, 0x00000aff, + 0x00000b13, 0x00000b27, 0x00000b3b, 0x00000b4f, 0x00000b63, + 0x00000b77, 0x00000b8b, 0x00000ba0, 0x00000bb4, 0x00000bc9, + 0x00000bde, 0x00000bf3, 0x00000c08, 0x00000c1d, 0x00000c32, + + 0x00000c47, 0x00000c5c, 0x00000c72, 0x00000c87, 0x00000c9d, + 0x00000cb3, 0x00000cc9, 0x00000cdf, 0x00000cf5, 0x00000d0b, + 0x00000d22, 0x00000d38, 0x00000d4f, 0x00000d65, 0x00000d7c, + 0x00000d93, 0x00000daa, 0x00000dc2, 0x00000dd9, 0x00000df0, + 0x00000e08, 0x00000e20, 0x00000e38, 0x00000e50, 0x00000e68, + 0x00000e80, 0x00000e99, 0x00000eb2, 0x00000eca, 0x00000ee3, + 0x00000efc, 0x00000f16, 0x00000f2f, 0x00000f49, 0x00000f62, + 0x00000f7c, 0x00000f96, 0x00000fb0, 0x00000fcb, 0x00000fe5, + + 0x00001000, 0x0000101b, 0x00001036, 0x00001051, 0x0000106d, + 0x00001088, 0x000010a4, 0x000010c0, 0x000010dc, 0x000010f9, + 0x00001115, 0x00001132, 0x0000114f, 0x0000116c, 0x0000118a, + 0x000011a7, 0x000011c5, 0x000011e3, 0x00001201, 0x00001220, + 0x0000123f, 0x0000125e, 0x0000127d, 0x0000129c, 0x000012bc, + 0x000012dc, 0x000012fc, 0x0000131c, 0x0000133d, 0x0000135e, + 0x0000137f, 0x000013a0, 0x000013c2, 0x000013e4, 0x00001406, + 0x00001429, 0x0000144c, 0x0000146f, 0x00001492, 0x000014b6, + + 0x000014da, 0x000014fe, 0x00001523, 0x00001548, 0x0000156d, + 0x00001593, 0x000015b9, 0x000015df, 0x00001606, 0x0000162d, + 0x00001654, 0x0000167c, 0x000016a4, 0x000016cc, 0x000016f5, + 0x0000171e, 0x00001748, 0x00001772, 0x0000179c, 0x000017c7, + 0x000017f2, 0x0000181e, 0x0000184a, 0x00001876, 0x000018a3, + 0x000018d1, 0x000018ff, 0x0000192d, 0x0000195c, 0x0000198b, + 0x000019bb, 0x000019eb, 0x00001a1c, 0x00001a4d, 0x00001a7f, + 0x00001ab2, 0x00001ae5, 0x00001b18, 0x00001b4c, 0x00001b81, + + 0x00001bb6, 0x00001bec, 0x00001c23, 0x00001c5a, 0x00001c92, + 0x00001cca, 0x00001d04, 0x00001d3d, 0x00001d78, 0x00001db3, + 0x00001def, 0x00001e2c, 0x00001e69, 0x00001ea7, 0x00001ee6, + 0x00001f26, 0x00001f67, 0x00001fa8, 0x00001feb, 0x0000202e, + 0x00002072, 0x000020b7, 0x000020fd, 0x00002144, 0x0000218b, + 0x000021d4, 0x0000221e, 0x00002269, 0x000022b5, 0x00002302, + 0x00002350, 0x0000239f, 0x000023f0, 0x00002441, 0x00002494, + 0x000024e8, 0x0000253e, 0x00002594, 0x000025ec, 0x00002646, + + 0x000026a1, 0x000026fd, 0x0000275b, 0x000027ba, 0x0000281b, + 0x0000287d, 0x000028e1, 0x00002947, 0x000029ae, 0x00002a18, + 0x00002a83, 0x00002af0, 0x00002b5f, 0x00002bd0, 0x00002c43, + 0x00002cb8, 0x00002d2f, 0x00002da8, 0x00002e24, 0x00002ea2, + 0x00002f22, 0x00002fa5, 0x0000302b, 0x000030b3, 0x0000313e, + 0x000031cc, 0x0000325d, 0x000032f0, 0x00003387, 0x00003421, + 0x000034bf, 0x00003560, 0x00003604, 0x000036ac, 0x00003758, + 0x00003807, 0x000038bb, 0x00003973, 0x00003a30, 0x00003af1, + + 0x00003bb6, 0x00003c81, 0x00003d51, 0x00003e26, 0x00003f00, + 0x00003fe0, 0x000040c6, 0x000041b2, 0x000042a5, 0x0000439e, + 0x0000449f, 0x000045a6, 0x000046b6, 0x000047cd, 0x000048ed, + 0x00004a15, 0x00004b46, 0x00004c81, 0x00004dc6, 0x00004f15, + 0x00005070, 0x000051d6, 0x00005348, 0x000054c7, 0x00005654, + 0x000057ef, 0x00005999, 0x00005b53, 0x00005d1d, 0x00005efa, + 0x000060e9, 0x000062ec, 0x00006505, 0x00006734, 0x0000697c, + 0x00006bdd, 0x00006e5a, 0x000070f4, 0x000073ad, 0x00007689, + + 0x00007988, 0x00007caf, 0x00008000, 0x0000837f, 0x0000872f, + 0x00008b15, 0x00008f35, 0x00009395, 0x0000983b, 0x00009d2d, + 0x0000a273, 0x0000a817, 0x0000ae20, 0x0000b49d, 0x0000bb98, + 0x0000c322, 0x0000cb4d, 0x0000d42d, 0x0000dddb, 0x0000e874, + 0x0000f41d, 0x000100ff, 0x00010f4f, 0x00011f4e, 0x0001314c, + 0x000145b0, 0x00015cfd, 0x000177de, 0x0001973a, 0x0001bc49, + 0x0001e8c0, 0x00021f18, 0x00026304, 0x0002ba57, 0x00032ec4, + 0x0003d1c2, 0x0004c63d, 0x00065dae, 0x00098c92, 0x0013193a, + + 0x00000000, 0xffece6ba, 0xfff6736b, 0xfff9a24b, 0xfffb39c2, + 0xfffc2e3d, 0xfffcd13b, 0xfffd45a9, 0xfffd9cfc, 0xfffde0e8, + 0xfffe1740, 0xfffe43b7, 0xfffe68c6, 0xfffe8822, 0xfffea303, + 0xfffeba50, 0xfffeceb4, 0xfffee0b2, 0xfffef0b1, 0xfffeff01, + 0xffff0be3, 0xffff178c, 0xffff2225, 0xffff2bd3, 0xffff34b3, + 0xffff3cde, 0xffff4468, 0xffff4b63, 0xffff51e0, 0xffff57e9, + 0xffff5d8d, 0xffff62d3, 0xffff67c5, 0xffff6c6b, 0xffff70cb, + 0xffff74eb, 0xffff78d1, 0xffff7c81, 0xffff8000, 0xffff8351, + + 0xffff8678, 0xffff8977, 0xffff8c53, 0xffff8f0c, 0xffff91a6, + 0xffff9423, 0xffff9684, 0xffff98cc, 0xffff9afb, 0xffff9d14, + 0xffff9f17, 0xffffa106, 0xffffa2e3, 0xffffa4ad, 0xffffa667, + 0xffffa811, 0xffffa9ac, 0xffffab39, 0xffffacb8, 0xffffae2a, + 0xffffaf90, 0xffffb0ea, 0xffffb23a, 0xffffb37f, 0xffffb4ba, + 0xffffb5eb, 0xffffb713, 0xffffb833, 0xffffb94a, 0xffffba5a, + 0xffffbb61, 0xffffbc62, 0xffffbd5b, 0xffffbe4e, 0xffffbf3a, + 0xffffc020, 0xffffc100, 0xffffc1da, 0xffffc2af, 0xffffc37f, + + 0xffffc44a, 0xffffc50f, 0xffffc5d0, 0xffffc68d, 0xffffc745, + 0xffffc7f9, 0xffffc8a8, 0xffffc954, 0xffffc9fc, 0xffffcaa0, + 0xffffcb41, 0xffffcbdf, 0xffffcc79, 0xffffcd10, 0xffffcda3, + 0xffffce34, 0xffffcec2, 0xffffcf4d, 0xffffcfd5, 0xffffd05b, + 0xffffd0de, 0xffffd15e, 0xffffd1dc, 0xffffd258, 0xffffd2d1, + 0xffffd348, 0xffffd3bd, 0xffffd430, 0xffffd4a1, 0xffffd510, + 0xffffd57d, 0xffffd5e8, 0xffffd652, 0xffffd6b9, 0xffffd71f, + 0xffffd783, 0xffffd7e5, 0xffffd846, 0xffffd8a5, 0xffffd903, + + 0xffffd95f, 0xffffd9ba, 0xffffda14, 0xffffda6c, 0xffffdac2, + 0xffffdb18, 0xffffdb6c, 0xffffdbbf, 0xffffdc10, 0xffffdc61, + 0xffffdcb0, 0xffffdcfe, 0xffffdd4b, 0xffffdd97, 0xffffdde2, + 0xffffde2c, 0xffffde75, 0xffffdebc, 0xffffdf03, 0xffffdf49, + 0xffffdf8e, 0xffffdfd2, 0xffffe015, 0xffffe058, 0xffffe099, + 0xffffe0da, 0xffffe11a, 0xffffe159, 0xffffe197, 0xffffe1d4, + 0xffffe211, 0xffffe24d, 0xffffe288, 0xffffe2c3, 0xffffe2fc, + 0xffffe336, 0xffffe36e, 0xffffe3a6, 0xffffe3dd, 0xffffe414, + + 0xffffe44a, 0xffffe47f, 0xffffe4b4, 0xffffe4e8, 0xffffe51b, + 0xffffe54e, 0xffffe581, 0xffffe5b3, 0xffffe5e4, 0xffffe615, + 0xffffe645, 0xffffe675, 0xffffe6a4, 0xffffe6d3, 0xffffe701, + 0xffffe72f, 0xffffe75d, 0xffffe78a, 0xffffe7b6, 0xffffe7e2, + 0xffffe80e, 0xffffe839, 0xffffe864, 0xffffe88e, 0xffffe8b8, + 0xffffe8e2, 0xffffe90b, 0xffffe934, 0xffffe95c, 0xffffe984, + 0xffffe9ac, 0xffffe9d3, 0xffffe9fa, 0xffffea21, 0xffffea47, + 0xffffea6d, 0xffffea93, 0xffffeab8, 0xffffeadd, 0xffffeb02, + + 0xffffeb26, 0xffffeb4a, 0xffffeb6e, 0xffffeb91, 0xffffebb4, + 0xffffebd7, 0xffffebfa, 0xffffec1c, 0xffffec3e, 0xffffec60, + 0xffffec81, 0xffffeca2, 0xffffecc3, 0xffffece4, 0xffffed04, + 0xffffed24, 0xffffed44, 0xffffed64, 0xffffed83, 0xffffeda2, + 0xffffedc1, 0xffffede0, 0xffffedff, 0xffffee1d, 0xffffee3b, + 0xffffee59, 0xffffee76, 0xffffee94, 0xffffeeb1, 0xffffeece, + 0xffffeeeb, 0xffffef07, 0xffffef24, 0xffffef40, 0xffffef5c, + 0xffffef78, 0xffffef93, 0xffffefaf, 0xffffefca, 0xffffefe5, + + 0xfffff000, 0xfffff01b, 0xfffff035, 0xfffff050, 0xfffff06a, + 0xfffff084, 0xfffff09e, 0xfffff0b7, 0xfffff0d1, 0xfffff0ea, + 0xfffff104, 0xfffff11d, 0xfffff136, 0xfffff14e, 0xfffff167, + 0xfffff180, 0xfffff198, 0xfffff1b0, 0xfffff1c8, 0xfffff1e0, + 0xfffff1f8, 0xfffff210, 0xfffff227, 0xfffff23e, 0xfffff256, + 0xfffff26d, 0xfffff284, 0xfffff29b, 0xfffff2b1, 0xfffff2c8, + 0xfffff2de, 0xfffff2f5, 0xfffff30b, 0xfffff321, 0xfffff337, + 0xfffff34d, 0xfffff363, 0xfffff379, 0xfffff38e, 0xfffff3a4, + + 0xfffff3b9, 0xfffff3ce, 0xfffff3e3, 0xfffff3f8, 0xfffff40d, + 0xfffff422, 0xfffff437, 0xfffff44c, 0xfffff460, 0xfffff475, + 0xfffff489, 0xfffff49d, 0xfffff4b1, 0xfffff4c5, 0xfffff4d9, + 0xfffff4ed, 0xfffff501, 0xfffff515, 0xfffff528, 0xfffff53c, + 0xfffff54f, 0xfffff562, 0xfffff576, 0xfffff589, 0xfffff59c, + 0xfffff5af, 0xfffff5c2, 0xfffff5d5, 0xfffff5e8, 0xfffff5fa, + 0xfffff60d, 0xfffff61f, 0xfffff632, 0xfffff644, 0xfffff657, + 0xfffff669, 0xfffff67b, 0xfffff68d, 0xfffff69f, 0xfffff6b1, + + 0xfffff6c3, 0xfffff6d5, 0xfffff6e7, 0xfffff6f8, 0xfffff70a, + 0xfffff71c, 0xfffff72d, 0xfffff73f, 0xfffff750, 0xfffff761, + 0xfffff773, 0xfffff784, 0xfffff795, 0xfffff7a6, 0xfffff7b7, + 0xfffff7c8, 0xfffff7d9, 0xfffff7ea, 0xfffff7fb, 0xfffff80b, + 0xfffff81c, 0xfffff82d, 0xfffff83d, 0xfffff84e, 0xfffff85e, + 0xfffff86f, 0xfffff87f, 0xfffff88f, 0xfffff8a0, 0xfffff8b0, + 0xfffff8c0, 0xfffff8d0, 0xfffff8e0, 0xfffff8f0, 0xfffff900, + 0xfffff910, 0xfffff920, 0xfffff930, 0xfffff940, 0xfffff950, + + 0xfffff95f, 0xfffff96f, 0xfffff97f, 0xfffff98e, 0xfffff99e, + 0xfffff9ad, 0xfffff9bd, 0xfffff9cc, 0xfffff9dc, 0xfffff9eb, + 0xfffff9fa, 0xfffffa0a, 0xfffffa19, 0xfffffa28, 0xfffffa37, + 0xfffffa46, 0xfffffa56, 0xfffffa65, 0xfffffa74, 0xfffffa83, + 0xfffffa92, 0xfffffaa1, 0xfffffaaf, 0xfffffabe, 0xfffffacd, + 0xfffffadc, 0xfffffaeb, 0xfffffaf9, 0xfffffb08, 0xfffffb17, + 0xfffffb25, 0xfffffb34, 0xfffffb43, 0xfffffb51, 0xfffffb60, + 0xfffffb6e, 0xfffffb7d, 0xfffffb8b, 0xfffffb9a, 0xfffffba8, + + 0xfffffbb6, 0xfffffbc5, 0xfffffbd3, 0xfffffbe1, 0xfffffbf0, + 0xfffffbfe, 0xfffffc0c, 0xfffffc1a, 0xfffffc29, 0xfffffc37, + 0xfffffc45, 0xfffffc53, 0xfffffc61, 0xfffffc6f, 0xfffffc7d, + 0xfffffc8b, 0xfffffc99, 0xfffffca7, 0xfffffcb5, 0xfffffcc3, + 0xfffffcd1, 0xfffffcdf, 0xfffffced, 0xfffffcfb, 0xfffffd09, + 0xfffffd17, 0xfffffd25, 0xfffffd32, 0xfffffd40, 0xfffffd4e, + 0xfffffd5c, 0xfffffd6a, 0xfffffd77, 0xfffffd85, 0xfffffd93, + 0xfffffda0, 0xfffffdae, 0xfffffdbc, 0xfffffdc9, 0xfffffdd7, + + 0xfffffde5, 0xfffffdf2, 0xfffffe00, 0xfffffe0e, 0xfffffe1b, + 0xfffffe29, 0xfffffe36, 0xfffffe44, 0xfffffe51, 0xfffffe5f, + 0xfffffe6d, 0xfffffe7a, 0xfffffe88, 0xfffffe95, 0xfffffea3, + 0xfffffeb0, 0xfffffebe, 0xfffffecb, 0xfffffed9, 0xfffffee6, + 0xfffffef4, 0xffffff01, 0xffffff0e, 0xffffff1c, 0xffffff29, + 0xffffff37, 0xffffff44, 0xffffff52, 0xffffff5f, 0xffffff6c, + 0xffffff7a, 0xffffff87, 0xffffff95, 0xffffffa2, 0xffffffb0, + 0xffffffbd, 0xffffffca, 0xffffffd8, 0xffffffe5, 0xfffffff3, + + 0x00000000, 0x0000000d, 0x0000001b, 0x00000028, 0x00000036, + 0x00000043, 0x00000050, 0x0000005e, 0x0000006b, 0x00000079, + 0x00000086, 0x00000094, 0x000000a1, 0x000000ae, 0x000000bc, + 0x000000c9, 0x000000d7, 0x000000e4, 0x000000f2, 0x000000ff, + 0x0000010c, 0x0000011a, 0x00000127, 0x00000135, 0x00000142, + 0x00000150, 0x0000015d, 0x0000016b, 0x00000178, 0x00000186, + 0x00000193, 0x000001a1, 0x000001af, 0x000001bc, 0x000001ca, + 0x000001d7, 0x000001e5, 0x000001f2, 0x00000200, 0x0000020e, + + 0x0000021b, 0x00000229, 0x00000237, 0x00000244, 0x00000252, + 0x00000260, 0x0000026d, 0x0000027b, 0x00000289, 0x00000296, + 0x000002a4, 0x000002b2, 0x000002c0, 0x000002ce, 0x000002db, + 0x000002e9, 0x000002f7, 0x00000305, 0x00000313, 0x00000321, + 0x0000032f, 0x0000033d, 0x0000034b, 0x00000359, 0x00000367, + 0x00000375, 0x00000383, 0x00000391, 0x0000039f, 0x000003ad, + 0x000003bb, 0x000003c9, 0x000003d7, 0x000003e6, 0x000003f4, + 0x00000402, 0x00000410, 0x0000041f, 0x0000042d, 0x0000043b, + + 0x0000044a, 0x00000458, 0x00000466, 0x00000475, 0x00000483, + 0x00000492, 0x000004a0, 0x000004af, 0x000004bd, 0x000004cc, + 0x000004db, 0x000004e9, 0x000004f8, 0x00000507, 0x00000515, + 0x00000524, 0x00000533, 0x00000542, 0x00000551, 0x0000055f, + 0x0000056e, 0x0000057d, 0x0000058c, 0x0000059b, 0x000005aa, + 0x000005ba, 0x000005c9, 0x000005d8, 0x000005e7, 0x000005f6, + 0x00000606, 0x00000615, 0x00000624, 0x00000634, 0x00000643, + 0x00000653, 0x00000662, 0x00000672, 0x00000681, 0x00000691, + + 0x000006a1, 0x000006b0, 0x000006c0, 0x000006d0, 0x000006e0, + 0x000006f0, 0x00000700, 0x00000710, 0x00000720, 0x00000730, + 0x00000740, 0x00000750, 0x00000760, 0x00000771, 0x00000781, + 0x00000791, 0x000007a2, 0x000007b2, 0x000007c3, 0x000007d3, + 0x000007e4, 0x000007f5, 0x00000805, 0x00000816, 0x00000827, + 0x00000838, 0x00000849, 0x0000085a, 0x0000086b, 0x0000087c, + 0x0000088d, 0x0000089f, 0x000008b0, 0x000008c1, 0x000008d3, + 0x000008e4, 0x000008f6, 0x00000908, 0x00000919, 0x0000092b, + + 0x0000093d, 0x0000094f, 0x00000961, 0x00000973, 0x00000985, + 0x00000997, 0x000009a9, 0x000009bc, 0x000009ce, 0x000009e1, + 0x000009f3, 0x00000a06, 0x00000a18, 0x00000a2b, 0x00000a3e, + 0x00000a51, 0x00000a64, 0x00000a77, 0x00000a8a, 0x00000a9e, + 0x00000ab1, 0x00000ac4, 0x00000ad8, 0x00000aeb, 0x00000aff, + 0x00000b13, 0x00000b27, 0x00000b3b, 0x00000b4f, 0x00000b63, + 0x00000b77, 0x00000b8b, 0x00000ba0, 0x00000bb4, 0x00000bc9, + 0x00000bde, 0x00000bf3, 0x00000c08, 0x00000c1d, 0x00000c32, + + 0x00000c47, 0x00000c5c, 0x00000c72, 0x00000c87, 0x00000c9d, + 0x00000cb3, 0x00000cc9, 0x00000cdf, 0x00000cf5, 0x00000d0b, + 0x00000d21, 0x00000d38, 0x00000d4f, 0x00000d65, 0x00000d7c, + 0x00000d93, 0x00000daa, 0x00000dc2, 0x00000dd9, 0x00000df0, + 0x00000e08, 0x00000e20, 0x00000e38, 0x00000e50, 0x00000e68, + 0x00000e80, 0x00000e99, 0x00000eb2, 0x00000eca, 0x00000ee3, + 0x00000efc, 0x00000f16, 0x00000f2f, 0x00000f49, 0x00000f62, + 0x00000f7c, 0x00000f96, 0x00000fb0, 0x00000fcb, 0x00000fe5, + + 0x00001000, 0x0000101b, 0x00001036, 0x00001051, 0x0000106d, + 0x00001088, 0x000010a4, 0x000010c0, 0x000010dc, 0x000010f9, + 0x00001115, 0x00001132, 0x0000114f, 0x0000116c, 0x0000118a, + 0x000011a7, 0x000011c5, 0x000011e3, 0x00001201, 0x00001220, + 0x0000123f, 0x0000125e, 0x0000127d, 0x0000129c, 0x000012bc, + 0x000012dc, 0x000012fc, 0x0000131c, 0x0000133d, 0x0000135e, + 0x0000137f, 0x000013a0, 0x000013c2, 0x000013e4, 0x00001406, + 0x00001429, 0x0000144c, 0x0000146f, 0x00001492, 0x000014b6, + + 0x000014da, 0x000014fe, 0x00001523, 0x00001548, 0x0000156d, + 0x00001593, 0x000015b9, 0x000015df, 0x00001606, 0x0000162d, + 0x00001654, 0x0000167c, 0x000016a4, 0x000016cc, 0x000016f5, + 0x0000171e, 0x00001748, 0x00001772, 0x0000179c, 0x000017c7, + 0x000017f2, 0x0000181e, 0x0000184a, 0x00001876, 0x000018a3, + 0x000018d1, 0x000018ff, 0x0000192d, 0x0000195c, 0x0000198b, + 0x000019bb, 0x000019eb, 0x00001a1c, 0x00001a4d, 0x00001a7f, + 0x00001ab2, 0x00001ae5, 0x00001b18, 0x00001b4c, 0x00001b81, + + 0x00001bb6, 0x00001bec, 0x00001c23, 0x00001c5a, 0x00001c92, + 0x00001cca, 0x00001d04, 0x00001d3d, 0x00001d78, 0x00001db3, + 0x00001def, 0x00001e2c, 0x00001e69, 0x00001ea7, 0x00001ee6, + 0x00001f26, 0x00001f67, 0x00001fa8, 0x00001feb, 0x0000202e, + 0x00002072, 0x000020b7, 0x000020fd, 0x00002144, 0x0000218b, + 0x000021d4, 0x0000221e, 0x00002269, 0x000022b5, 0x00002302, + 0x00002350, 0x0000239f, 0x000023f0, 0x00002441, 0x00002494, + 0x000024e8, 0x0000253e, 0x00002594, 0x000025ec, 0x00002646, + + 0x000026a1, 0x000026fd, 0x0000275b, 0x000027ba, 0x0000281b, + 0x0000287d, 0x000028e1, 0x00002947, 0x000029ae, 0x00002a18, + 0x00002a83, 0x00002af0, 0x00002b5f, 0x00002bd0, 0x00002c43, + 0x00002cb8, 0x00002d2f, 0x00002da8, 0x00002e24, 0x00002ea2, + 0x00002f22, 0x00002fa5, 0x0000302b, 0x000030b3, 0x0000313e, + 0x000031cc, 0x0000325d, 0x000032f0, 0x00003387, 0x00003421, + 0x000034bf, 0x0000355f, 0x00003604, 0x000036ac, 0x00003758, + 0x00003807, 0x000038bb, 0x00003973, 0x00003a30, 0x00003af1, + + 0x00003bb6, 0x00003c81, 0x00003d51, 0x00003e26, 0x00003f00, + 0x00003fe0, 0x000040c6, 0x000041b2, 0x000042a5, 0x0000439e, + 0x0000449f, 0x000045a6, 0x000046b6, 0x000047cd, 0x000048ec, + 0x00004a15, 0x00004b46, 0x00004c81, 0x00004dc6, 0x00004f15, + 0x00005070, 0x000051d6, 0x00005348, 0x000054c7, 0x00005654, + 0x000057ef, 0x00005999, 0x00005b53, 0x00005d1d, 0x00005efa, + 0x000060e9, 0x000062ec, 0x00006505, 0x00006734, 0x0000697c, + 0x00006bdd, 0x00006e5a, 0x000070f4, 0x000073ad, 0x00007688, + + 0x00007988, 0x00007caf, 0x00008000, 0x0000837f, 0x0000872f, + 0x00008b15, 0x00008f35, 0x00009395, 0x0000983b, 0x00009d2d, + 0x0000a273, 0x0000a816, 0x0000ae20, 0x0000b49c, 0x0000bb98, + 0x0000c322, 0x0000cb4d, 0x0000d42d, 0x0000dddb, 0x0000e874, + 0x0000f41d, 0x000100ff, 0x00010f4f, 0x00011f4e, 0x0001314c, + 0x000145b0, 0x00015cfd, 0x000177de, 0x0001973a, 0x0001bc49, + 0x0001e8c0, 0x00021f18, 0x00026302, 0x0002ba55, 0x00032ec2, + 0x0003d1c0, 0x0004c63a, 0x00065dad, 0x00098c8f, 0x0013192e, + + 0x00000000 +}; + +/* This is SIN lookup table. NOTE: + * 1. The index ranges in value from 0 to 3*PI/2-1. + * 2. The value obtain from the table have 12 bits of fraction. + * 3. COS(x) = SIN(x + HALFPI) + */ + +const int16_t g_sin_table[TWOPI + HALFPI + 1] = +{ + 0x0000, 0x000d, 0x001b, 0x0028, 0x0036, 0x0043, 0x0050, 0x005e, + 0x006b, 0x0079, 0x0086, 0x0093, 0x00a1, 0x00ae, 0x00bc, 0x00c9, + 0x00d6, 0x00e4, 0x00f1, 0x00ff, 0x010c, 0x0119, 0x0127, 0x0134, + 0x0141, 0x014f, 0x015c, 0x0169, 0x0177, 0x0184, 0x0191, 0x019f, + 0x01ac, 0x01b9, 0x01c7, 0x01d4, 0x01e1, 0x01ef, 0x01fc, 0x0209, + 0x0217, 0x0224, 0x0231, 0x023e, 0x024c, 0x0259, 0x0266, 0x0274, + 0x0281, 0x028e, 0x029b, 0x02a8, 0x02b6, 0x02c3, 0x02d0, 0x02dd, + 0x02ea, 0x02f8, 0x0305, 0x0312, 0x031f, 0x032c, 0x0339, 0x0346, + + 0x0354, 0x0361, 0x036e, 0x037b, 0x0388, 0x0395, 0x03a2, 0x03af, + 0x03bc, 0x03c9, 0x03d6, 0x03e3, 0x03f0, 0x03fd, 0x040a, 0x0417, + 0x0424, 0x0431, 0x043e, 0x044b, 0x0458, 0x0465, 0x0472, 0x047e, + 0x048b, 0x0498, 0x04a5, 0x04b2, 0x04bf, 0x04cb, 0x04d8, 0x04e5, + 0x04f2, 0x04fe, 0x050b, 0x0518, 0x0525, 0x0531, 0x053e, 0x054b, + 0x0557, 0x0564, 0x0571, 0x057d, 0x058a, 0x0596, 0x05a3, 0x05af, + 0x05bc, 0x05c8, 0x05d5, 0x05e1, 0x05ee, 0x05fa, 0x0607, 0x0613, + 0x061f, 0x062c, 0x0638, 0x0645, 0x0651, 0x065d, 0x0669, 0x0676, + + 0x0682, 0x068e, 0x069a, 0x06a7, 0x06b3, 0x06bf, 0x06cb, 0x06d7, + 0x06e3, 0x06ef, 0x06fc, 0x0708, 0x0714, 0x0720, 0x072c, 0x0738, + 0x0744, 0x074f, 0x075b, 0x0767, 0x0773, 0x077f, 0x078b, 0x0797, + 0x07a2, 0x07ae, 0x07ba, 0x07c6, 0x07d1, 0x07dd, 0x07e9, 0x07f4, + 0x0800, 0x080c, 0x0817, 0x0823, 0x082e, 0x083a, 0x0845, 0x0851, + 0x085c, 0x0868, 0x0873, 0x087e, 0x088a, 0x0895, 0x08a0, 0x08ac, + 0x08b7, 0x08c2, 0x08cd, 0x08d8, 0x08e4, 0x08ef, 0x08fa, 0x0905, + 0x0910, 0x091b, 0x0926, 0x0931, 0x093c, 0x0947, 0x0952, 0x095d, + + 0x0968, 0x0972, 0x097d, 0x0988, 0x0993, 0x099d, 0x09a8, 0x09b3, + 0x09bd, 0x09c8, 0x09d3, 0x09dd, 0x09e8, 0x09f2, 0x09fd, 0x0a07, + 0x0a12, 0x0a1c, 0x0a26, 0x0a31, 0x0a3b, 0x0a45, 0x0a50, 0x0a5a, + 0x0a64, 0x0a6e, 0x0a78, 0x0a83, 0x0a8d, 0x0a97, 0x0aa1, 0x0aab, + 0x0ab5, 0x0abf, 0x0ac9, 0x0ad3, 0x0adc, 0x0ae6, 0x0af0, 0x0afa, + 0x0b04, 0x0b0d, 0x0b17, 0x0b21, 0x0b2a, 0x0b34, 0x0b3d, 0x0b47, + 0x0b50, 0x0b5a, 0x0b63, 0x0b6d, 0x0b76, 0x0b7f, 0x0b89, 0x0b92, + 0x0b9b, 0x0ba4, 0x0bae, 0x0bb7, 0x0bc0, 0x0bc9, 0x0bd2, 0x0bdb, + + 0x0be4, 0x0bed, 0x0bf6, 0x0bff, 0x0c08, 0x0c10, 0x0c19, 0x0c22, + 0x0c2b, 0x0c33, 0x0c3c, 0x0c45, 0x0c4d, 0x0c56, 0x0c5e, 0x0c67, + 0x0c6f, 0x0c78, 0x0c80, 0x0c88, 0x0c91, 0x0c99, 0x0ca1, 0x0ca9, + 0x0cb2, 0x0cba, 0x0cc2, 0x0cca, 0x0cd2, 0x0cda, 0x0ce2, 0x0cea, + 0x0cf2, 0x0cfa, 0x0d01, 0x0d09, 0x0d11, 0x0d19, 0x0d20, 0x0d28, + 0x0d30, 0x0d37, 0x0d3f, 0x0d46, 0x0d4e, 0x0d55, 0x0d5d, 0x0d64, + 0x0d6b, 0x0d72, 0x0d7a, 0x0d81, 0x0d88, 0x0d8f, 0x0d96, 0x0d9d, + 0x0da4, 0x0dab, 0x0db2, 0x0db9, 0x0dc0, 0x0dc7, 0x0dce, 0x0dd5, + + 0x0ddb, 0x0de2, 0x0de9, 0x0def, 0x0df6, 0x0dfc, 0x0e03, 0x0e09, + 0x0e10, 0x0e16, 0x0e1c, 0x0e23, 0x0e29, 0x0e2f, 0x0e35, 0x0e3b, + 0x0e42, 0x0e48, 0x0e4e, 0x0e54, 0x0e5a, 0x0e5f, 0x0e65, 0x0e6b, + 0x0e71, 0x0e77, 0x0e7c, 0x0e82, 0x0e88, 0x0e8d, 0x0e93, 0x0e98, + 0x0e9e, 0x0ea3, 0x0ea9, 0x0eae, 0x0eb3, 0x0eb9, 0x0ebe, 0x0ec3, + 0x0ec8, 0x0ecd, 0x0ed2, 0x0ed7, 0x0edc, 0x0ee1, 0x0ee6, 0x0eeb, + 0x0ef0, 0x0ef5, 0x0ef9, 0x0efe, 0x0f03, 0x0f07, 0x0f0c, 0x0f11, + 0x0f15, 0x0f1a, 0x0f1e, 0x0f22, 0x0f27, 0x0f2b, 0x0f2f, 0x0f33, + + 0x0f38, 0x0f3c, 0x0f40, 0x0f44, 0x0f48, 0x0f4c, 0x0f50, 0x0f53, + 0x0f57, 0x0f5b, 0x0f5f, 0x0f63, 0x0f66, 0x0f6a, 0x0f6d, 0x0f71, + 0x0f74, 0x0f78, 0x0f7b, 0x0f7f, 0x0f82, 0x0f85, 0x0f88, 0x0f8c, + 0x0f8f, 0x0f92, 0x0f95, 0x0f98, 0x0f9b, 0x0f9e, 0x0fa1, 0x0fa4, + 0x0fa6, 0x0fa9, 0x0fac, 0x0faf, 0x0fb1, 0x0fb4, 0x0fb6, 0x0fb9, + 0x0fbb, 0x0fbe, 0x0fc0, 0x0fc3, 0x0fc5, 0x0fc7, 0x0fc9, 0x0fcb, + 0x0fce, 0x0fd0, 0x0fd2, 0x0fd4, 0x0fd6, 0x0fd8, 0x0fd9, 0x0fdb, + 0x0fdd, 0x0fdf, 0x0fe0, 0x0fe2, 0x0fe4, 0x0fe5, 0x0fe7, 0x0fe8, + + 0x0fea, 0x0feb, 0x0fec, 0x0fee, 0x0fef, 0x0ff0, 0x0ff1, 0x0ff2, + 0x0ff3, 0x0ff4, 0x0ff5, 0x0ff6, 0x0ff7, 0x0ff8, 0x0ff9, 0x0ffa, + 0x0ffa, 0x0ffb, 0x0ffc, 0x0ffc, 0x0ffd, 0x0ffd, 0x0ffe, 0x0ffe, + 0x0fff, 0x0fff, 0x0fff, 0x0fff, 0x1000, 0x1000, 0x1000, 0x1000, + 0x1000, 0x1000, 0x1000, 0x1000, 0x1000, 0x0fff, 0x0fff, 0x0fff, + 0x0fff, 0x0ffe, 0x0ffe, 0x0ffd, 0x0ffd, 0x0ffc, 0x0ffc, 0x0ffb, + 0x0ffa, 0x0ffa, 0x0ff9, 0x0ff8, 0x0ff7, 0x0ff6, 0x0ff5, 0x0ff4, + 0x0ff3, 0x0ff2, 0x0ff1, 0x0ff0, 0x0fef, 0x0fee, 0x0fec, 0x0feb, + + 0x0fea, 0x0fe8, 0x0fe7, 0x0fe5, 0x0fe4, 0x0fe2, 0x0fe0, 0x0fdf, + 0x0fdd, 0x0fdb, 0x0fd9, 0x0fd8, 0x0fd6, 0x0fd4, 0x0fd2, 0x0fd0, + 0x0fce, 0x0fcb, 0x0fc9, 0x0fc7, 0x0fc5, 0x0fc3, 0x0fc0, 0x0fbe, + 0x0fbb, 0x0fb9, 0x0fb6, 0x0fb4, 0x0fb1, 0x0faf, 0x0fac, 0x0fa9, + 0x0fa6, 0x0fa4, 0x0fa1, 0x0f9e, 0x0f9b, 0x0f98, 0x0f95, 0x0f92, + 0x0f8f, 0x0f8c, 0x0f88, 0x0f85, 0x0f82, 0x0f7f, 0x0f7b, 0x0f78, + 0x0f74, 0x0f71, 0x0f6d, 0x0f6a, 0x0f66, 0x0f63, 0x0f5f, 0x0f5b, + 0x0f57, 0x0f53, 0x0f50, 0x0f4c, 0x0f48, 0x0f44, 0x0f40, 0x0f3c, + + 0x0f38, 0x0f33, 0x0f2f, 0x0f2b, 0x0f27, 0x0f22, 0x0f1e, 0x0f1a, + 0x0f15, 0x0f11, 0x0f0c, 0x0f07, 0x0f03, 0x0efe, 0x0ef9, 0x0ef5, + 0x0ef0, 0x0eeb, 0x0ee6, 0x0ee1, 0x0edc, 0x0ed7, 0x0ed2, 0x0ecd, + 0x0ec8, 0x0ec3, 0x0ebe, 0x0eb9, 0x0eb3, 0x0eae, 0x0ea9, 0x0ea3, + 0x0e9e, 0x0e98, 0x0e93, 0x0e8d, 0x0e88, 0x0e82, 0x0e7c, 0x0e77, + 0x0e71, 0x0e6b, 0x0e65, 0x0e5f, 0x0e5a, 0x0e54, 0x0e4e, 0x0e48, + 0x0e42, 0x0e3b, 0x0e35, 0x0e2f, 0x0e29, 0x0e23, 0x0e1c, 0x0e16, + 0x0e10, 0x0e09, 0x0e03, 0x0dfc, 0x0df6, 0x0def, 0x0de9, 0x0de2, + + 0x0ddb, 0x0dd5, 0x0dce, 0x0dc7, 0x0dc0, 0x0db9, 0x0db2, 0x0dab, + 0x0da4, 0x0d9d, 0x0d96, 0x0d8f, 0x0d88, 0x0d81, 0x0d7a, 0x0d72, + 0x0d6b, 0x0d64, 0x0d5d, 0x0d55, 0x0d4e, 0x0d46, 0x0d3f, 0x0d37, + 0x0d30, 0x0d28, 0x0d20, 0x0d19, 0x0d11, 0x0d09, 0x0d01, 0x0cfa, + 0x0cf2, 0x0cea, 0x0ce2, 0x0cda, 0x0cd2, 0x0cca, 0x0cc2, 0x0cba, + 0x0cb2, 0x0ca9, 0x0ca1, 0x0c99, 0x0c91, 0x0c88, 0x0c80, 0x0c78, + 0x0c6f, 0x0c67, 0x0c5e, 0x0c56, 0x0c4d, 0x0c45, 0x0c3c, 0x0c33, + 0x0c2b, 0x0c22, 0x0c19, 0x0c10, 0x0c08, 0x0bff, 0x0bf6, 0x0bed, + + 0x0be4, 0x0bdb, 0x0bd2, 0x0bc9, 0x0bc0, 0x0bb7, 0x0bae, 0x0ba4, + 0x0b9b, 0x0b92, 0x0b89, 0x0b7f, 0x0b76, 0x0b6d, 0x0b63, 0x0b5a, + 0x0b50, 0x0b47, 0x0b3d, 0x0b34, 0x0b2a, 0x0b21, 0x0b17, 0x0b0d, + 0x0b04, 0x0afa, 0x0af0, 0x0ae6, 0x0adc, 0x0ad3, 0x0ac9, 0x0abf, + 0x0ab5, 0x0aab, 0x0aa1, 0x0a97, 0x0a8d, 0x0a83, 0x0a78, 0x0a6e, + 0x0a64, 0x0a5a, 0x0a50, 0x0a45, 0x0a3b, 0x0a31, 0x0a26, 0x0a1c, + 0x0a12, 0x0a07, 0x09fd, 0x09f2, 0x09e8, 0x09dd, 0x09d3, 0x09c8, + 0x09bd, 0x09b3, 0x09a8, 0x099d, 0x0993, 0x0988, 0x097d, 0x0972, + + 0x0968, 0x095d, 0x0952, 0x0947, 0x093c, 0x0931, 0x0926, 0x091b, + 0x0910, 0x0905, 0x08fa, 0x08ef, 0x08e4, 0x08d8, 0x08cd, 0x08c2, + 0x08b7, 0x08ac, 0x08a0, 0x0895, 0x088a, 0x087e, 0x0873, 0x0868, + 0x085c, 0x0851, 0x0845, 0x083a, 0x082e, 0x0823, 0x0817, 0x080c, + 0x0800, 0x07f4, 0x07e9, 0x07dd, 0x07d1, 0x07c6, 0x07ba, 0x07ae, + 0x07a2, 0x0797, 0x078b, 0x077f, 0x0773, 0x0767, 0x075b, 0x074f, + 0x0744, 0x0738, 0x072c, 0x0720, 0x0714, 0x0708, 0x06fc, 0x06ef, + 0x06e3, 0x06d7, 0x06cb, 0x06bf, 0x06b3, 0x06a7, 0x069a, 0x068e, + + 0x0682, 0x0676, 0x0669, 0x065d, 0x0651, 0x0645, 0x0638, 0x062c, + 0x061f, 0x0613, 0x0607, 0x05fa, 0x05ee, 0x05e1, 0x05d5, 0x05c8, + 0x05bc, 0x05af, 0x05a3, 0x0596, 0x058a, 0x057d, 0x0571, 0x0564, + 0x0557, 0x054b, 0x053e, 0x0531, 0x0525, 0x0518, 0x050b, 0x04fe, + 0x04f2, 0x04e5, 0x04d8, 0x04cb, 0x04bf, 0x04b2, 0x04a5, 0x0498, + 0x048b, 0x047e, 0x0472, 0x0465, 0x0458, 0x044b, 0x043e, 0x0431, + 0x0424, 0x0417, 0x040a, 0x03fd, 0x03f0, 0x03e3, 0x03d6, 0x03c9, + 0x03bc, 0x03af, 0x03a2, 0x0395, 0x0388, 0x037b, 0x036e, 0x0361, + + 0x0354, 0x0346, 0x0339, 0x032c, 0x031f, 0x0312, 0x0305, 0x02f8, + 0x02ea, 0x02dd, 0x02d0, 0x02c3, 0x02b6, 0x02a8, 0x029b, 0x028e, + 0x0281, 0x0274, 0x0266, 0x0259, 0x024c, 0x023e, 0x0231, 0x0224, + 0x0217, 0x0209, 0x01fc, 0x01ef, 0x01e1, 0x01d4, 0x01c7, 0x01b9, + 0x01ac, 0x019f, 0x0191, 0x0184, 0x0177, 0x0169, 0x015c, 0x014f, + 0x0141, 0x0134, 0x0127, 0x0119, 0x010c, 0x00ff, 0x00f1, 0x00e4, + 0x00d6, 0x00c9, 0x00bc, 0x00ae, 0x00a1, 0x0093, 0x0086, 0x0079, + 0x006b, 0x005e, 0x0050, 0x0043, 0x0036, 0x0028, 0x001b, 0x000d, + + 0x0000, 0xfff3, 0xffe5, 0xffd8, 0xffca, 0xffbd, 0xffb0, 0xffa2, + 0xff95, 0xff87, 0xff7a, 0xff6d, 0xff5f, 0xff52, 0xff44, 0xff37, + 0xff2a, 0xff1c, 0xff0f, 0xff01, 0xfef4, 0xfee7, 0xfed9, 0xfecc, + 0xfebf, 0xfeb1, 0xfea4, 0xfe97, 0xfe89, 0xfe7c, 0xfe6f, 0xfe61, + 0xfe54, 0xfe47, 0xfe39, 0xfe2c, 0xfe1f, 0xfe11, 0xfe04, 0xfdf7, + 0xfde9, 0xfddc, 0xfdcf, 0xfdc2, 0xfdb4, 0xfda7, 0xfd9a, 0xfd8c, + 0xfd7f, 0xfd72, 0xfd65, 0xfd58, 0xfd4a, 0xfd3d, 0xfd30, 0xfd23, + 0xfd16, 0xfd08, 0xfcfb, 0xfcee, 0xfce1, 0xfcd4, 0xfcc7, 0xfcba, + + 0xfcac, 0xfc9f, 0xfc92, 0xfc85, 0xfc78, 0xfc6b, 0xfc5e, 0xfc51, + 0xfc44, 0xfc37, 0xfc2a, 0xfc1d, 0xfc10, 0xfc03, 0xfbf6, 0xfbe9, + 0xfbdc, 0xfbcf, 0xfbc2, 0xfbb5, 0xfba8, 0xfb9b, 0xfb8e, 0xfb82, + 0xfb75, 0xfb68, 0xfb5b, 0xfb4e, 0xfb41, 0xfb35, 0xfb28, 0xfb1b, + 0xfb0e, 0xfb02, 0xfaf5, 0xfae8, 0xfadb, 0xfacf, 0xfac2, 0xfab5, + 0xfaa9, 0xfa9c, 0xfa8f, 0xfa83, 0xfa76, 0xfa6a, 0xfa5d, 0xfa51, + 0xfa44, 0xfa38, 0xfa2b, 0xfa1f, 0xfa12, 0xfa06, 0xf9f9, 0xf9ed, + 0xf9e1, 0xf9d4, 0xf9c8, 0xf9bb, 0xf9af, 0xf9a3, 0xf997, 0xf98a, + + 0xf97e, 0xf972, 0xf966, 0xf959, 0xf94d, 0xf941, 0xf935, 0xf929, + 0xf91d, 0xf911, 0xf904, 0xf8f8, 0xf8ec, 0xf8e0, 0xf8d4, 0xf8c8, + 0xf8bc, 0xf8b1, 0xf8a5, 0xf899, 0xf88d, 0xf881, 0xf875, 0xf869, + 0xf85e, 0xf852, 0xf846, 0xf83a, 0xf82f, 0xf823, 0xf817, 0xf80c, + 0xf800, 0xf7f4, 0xf7e9, 0xf7dd, 0xf7d2, 0xf7c6, 0xf7bb, 0xf7af, + 0xf7a4, 0xf798, 0xf78d, 0xf782, 0xf776, 0xf76b, 0xf760, 0xf754, + 0xf749, 0xf73e, 0xf733, 0xf728, 0xf71c, 0xf711, 0xf706, 0xf6fb, + 0xf6f0, 0xf6e5, 0xf6da, 0xf6cf, 0xf6c4, 0xf6b9, 0xf6ae, 0xf6a3, + + 0xf698, 0xf68e, 0xf683, 0xf678, 0xf66d, 0xf663, 0xf658, 0xf64d, + 0xf643, 0xf638, 0xf62d, 0xf623, 0xf618, 0xf60e, 0xf603, 0xf5f9, + 0xf5ee, 0xf5e4, 0xf5da, 0xf5cf, 0xf5c5, 0xf5bb, 0xf5b0, 0xf5a6, + 0xf59c, 0xf592, 0xf588, 0xf57d, 0xf573, 0xf569, 0xf55f, 0xf555, + 0xf54b, 0xf541, 0xf537, 0xf52d, 0xf524, 0xf51a, 0xf510, 0xf506, + 0xf4fd, 0xf4f3, 0xf4e9, 0xf4df, 0xf4d6, 0xf4cc, 0xf4c3, 0xf4b9, + 0xf4b0, 0xf4a6, 0xf49d, 0xf493, 0xf48a, 0xf481, 0xf477, 0xf46e, + 0xf465, 0xf45c, 0xf452, 0xf449, 0xf440, 0xf437, 0xf42e, 0xf425, + + 0xf41c, 0xf413, 0xf40a, 0xf401, 0xf3f8, 0xf3f0, 0xf3e7, 0xf3de, + 0xf3d5, 0xf3cd, 0xf3c4, 0xf3bb, 0xf3b3, 0xf3aa, 0xf3a2, 0xf399, + 0xf391, 0xf388, 0xf380, 0xf378, 0xf36f, 0xf367, 0xf35f, 0xf357, + 0xf34e, 0xf346, 0xf33e, 0xf336, 0xf32e, 0xf326, 0xf31e, 0xf316, + 0xf30e, 0xf306, 0xf2ff, 0xf2f7, 0xf2ef, 0xf2e7, 0xf2e0, 0xf2d8, + 0xf2d0, 0xf2c9, 0xf2c1, 0xf2ba, 0xf2b2, 0xf2ab, 0xf2a3, 0xf29c, + 0xf295, 0xf28e, 0xf286, 0xf27f, 0xf278, 0xf271, 0xf26a, 0xf263, + 0xf25c, 0xf255, 0xf24e, 0xf247, 0xf240, 0xf239, 0xf232, 0xf22b, + + 0xf225, 0xf21e, 0xf217, 0xf211, 0xf20a, 0xf204, 0xf1fd, 0xf1f7, + 0xf1f0, 0xf1ea, 0xf1e4, 0xf1dd, 0xf1d7, 0xf1d1, 0xf1cb, 0xf1c5, + 0xf1be, 0xf1b8, 0xf1b2, 0xf1ac, 0xf1a6, 0xf1a1, 0xf19b, 0xf195, + 0xf18f, 0xf189, 0xf184, 0xf17e, 0xf178, 0xf173, 0xf16d, 0xf168, + 0xf162, 0xf15d, 0xf157, 0xf152, 0xf14d, 0xf147, 0xf142, 0xf13d, + 0xf138, 0xf133, 0xf12e, 0xf129, 0xf124, 0xf11f, 0xf11a, 0xf115, + 0xf110, 0xf10b, 0xf107, 0xf102, 0xf0fd, 0xf0f9, 0xf0f4, 0xf0ef, + 0xf0eb, 0xf0e6, 0xf0e2, 0xf0de, 0xf0d9, 0xf0d5, 0xf0d1, 0xf0cd, + + 0xf0c8, 0xf0c4, 0xf0c0, 0xf0bc, 0xf0b8, 0xf0b4, 0xf0b0, 0xf0ad, + 0xf0a9, 0xf0a5, 0xf0a1, 0xf09d, 0xf09a, 0xf096, 0xf093, 0xf08f, + 0xf08c, 0xf088, 0xf085, 0xf081, 0xf07e, 0xf07b, 0xf078, 0xf074, + 0xf071, 0xf06e, 0xf06b, 0xf068, 0xf065, 0xf062, 0xf05f, 0xf05c, + 0xf05a, 0xf057, 0xf054, 0xf051, 0xf04f, 0xf04c, 0xf04a, 0xf047, + 0xf045, 0xf042, 0xf040, 0xf03d, 0xf03b, 0xf039, 0xf037, 0xf035, + 0xf032, 0xf030, 0xf02e, 0xf02c, 0xf02a, 0xf028, 0xf027, 0xf025, + 0xf023, 0xf021, 0xf020, 0xf01e, 0xf01c, 0xf01b, 0xf019, 0xf018, + + 0xf016, 0xf015, 0xf014, 0xf012, 0xf011, 0xf010, 0xf00f, 0xf00e, + 0xf00d, 0xf00c, 0xf00b, 0xf00a, 0xf009, 0xf008, 0xf007, 0xf006, + 0xf006, 0xf005, 0xf004, 0xf004, 0xf003, 0xf003, 0xf002, 0xf002, + 0xf001, 0xf001, 0xf001, 0xf001, 0xf000, 0xf000, 0xf000, 0xf000, + 0xf000, 0xf000, 0xf000, 0xf000, 0xf000, 0xf001, 0xf001, 0xf001, + 0xf001, 0xf002, 0xf002, 0xf003, 0xf003, 0xf004, 0xf004, 0xf005, + 0xf006, 0xf006, 0xf007, 0xf008, 0xf009, 0xf00a, 0xf00b, 0xf00c, + 0xf00d, 0xf00e, 0xf00f, 0xf010, 0xf011, 0xf012, 0xf014, 0xf015, + + 0xf016, 0xf018, 0xf019, 0xf01b, 0xf01c, 0xf01e, 0xf020, 0xf021, + 0xf023, 0xf025, 0xf027, 0xf028, 0xf02a, 0xf02c, 0xf02e, 0xf030, + 0xf032, 0xf035, 0xf037, 0xf039, 0xf03b, 0xf03d, 0xf040, 0xf042, + 0xf045, 0xf047, 0xf04a, 0xf04c, 0xf04f, 0xf051, 0xf054, 0xf057, + 0xf05a, 0xf05c, 0xf05f, 0xf062, 0xf065, 0xf068, 0xf06b, 0xf06e, + 0xf071, 0xf074, 0xf078, 0xf07b, 0xf07e, 0xf081, 0xf085, 0xf088, + 0xf08c, 0xf08f, 0xf093, 0xf096, 0xf09a, 0xf09d, 0xf0a1, 0xf0a5, + 0xf0a9, 0xf0ad, 0xf0b0, 0xf0b4, 0xf0b8, 0xf0bc, 0xf0c0, 0xf0c4, + + 0xf0c8, 0xf0cd, 0xf0d1, 0xf0d5, 0xf0d9, 0xf0de, 0xf0e2, 0xf0e6, + 0xf0eb, 0xf0ef, 0xf0f4, 0xf0f9, 0xf0fd, 0xf102, 0xf107, 0xf10b, + 0xf110, 0xf115, 0xf11a, 0xf11f, 0xf124, 0xf129, 0xf12e, 0xf133, + 0xf138, 0xf13d, 0xf142, 0xf147, 0xf14d, 0xf152, 0xf157, 0xf15d, + 0xf162, 0xf168, 0xf16d, 0xf173, 0xf178, 0xf17e, 0xf184, 0xf189, + 0xf18f, 0xf195, 0xf19b, 0xf1a1, 0xf1a6, 0xf1ac, 0xf1b2, 0xf1b8, + 0xf1be, 0xf1c5, 0xf1cb, 0xf1d1, 0xf1d7, 0xf1dd, 0xf1e4, 0xf1ea, + 0xf1f0, 0xf1f7, 0xf1fd, 0xf204, 0xf20a, 0xf211, 0xf217, 0xf21e, + + 0xf225, 0xf22b, 0xf232, 0xf239, 0xf240, 0xf247, 0xf24e, 0xf255, + 0xf25c, 0xf263, 0xf26a, 0xf271, 0xf278, 0xf27f, 0xf286, 0xf28e, + 0xf295, 0xf29c, 0xf2a3, 0xf2ab, 0xf2b2, 0xf2ba, 0xf2c1, 0xf2c9, + 0xf2d0, 0xf2d8, 0xf2e0, 0xf2e7, 0xf2ef, 0xf2f7, 0xf2ff, 0xf306, + 0xf30e, 0xf316, 0xf31e, 0xf326, 0xf32e, 0xf336, 0xf33e, 0xf346, + 0xf34e, 0xf357, 0xf35f, 0xf367, 0xf36f, 0xf378, 0xf380, 0xf388, + 0xf391, 0xf399, 0xf3a2, 0xf3aa, 0xf3b3, 0xf3bb, 0xf3c4, 0xf3cd, + 0xf3d5, 0xf3de, 0xf3e7, 0xf3f0, 0xf3f8, 0xf401, 0xf40a, 0xf413, + + 0xf41c, 0xf425, 0xf42e, 0xf437, 0xf440, 0xf449, 0xf452, 0xf45c, + 0xf465, 0xf46e, 0xf477, 0xf481, 0xf48a, 0xf493, 0xf49d, 0xf4a6, + 0xf4b0, 0xf4b9, 0xf4c3, 0xf4cc, 0xf4d6, 0xf4df, 0xf4e9, 0xf4f3, + 0xf4fc, 0xf506, 0xf510, 0xf51a, 0xf524, 0xf52d, 0xf537, 0xf541, + 0xf54b, 0xf555, 0xf55f, 0xf569, 0xf573, 0xf57d, 0xf588, 0xf592, + 0xf59c, 0xf5a6, 0xf5b0, 0xf5bb, 0xf5c5, 0xf5cf, 0xf5da, 0xf5e4, + 0xf5ee, 0xf5f9, 0xf603, 0xf60e, 0xf618, 0xf623, 0xf62d, 0xf638, + 0xf643, 0xf64d, 0xf658, 0xf663, 0xf66d, 0xf678, 0xf683, 0xf68e, + + 0xf698, 0xf6a3, 0xf6ae, 0xf6b9, 0xf6c4, 0xf6cf, 0xf6da, 0xf6e5, + 0xf6f0, 0xf6fb, 0xf706, 0xf711, 0xf71c, 0xf728, 0xf733, 0xf73e, + 0xf749, 0xf754, 0xf760, 0xf76b, 0xf776, 0xf782, 0xf78d, 0xf798, + 0xf7a4, 0xf7af, 0xf7bb, 0xf7c6, 0xf7d2, 0xf7dd, 0xf7e9, 0xf7f4, + 0xf800, 0xf80c, 0xf817, 0xf823, 0xf82f, 0xf83a, 0xf846, 0xf852, + 0xf85e, 0xf869, 0xf875, 0xf881, 0xf88d, 0xf899, 0xf8a5, 0xf8b1, + 0xf8bc, 0xf8c8, 0xf8d4, 0xf8e0, 0xf8ec, 0xf8f8, 0xf904, 0xf911, + 0xf91d, 0xf929, 0xf935, 0xf941, 0xf94d, 0xf959, 0xf966, 0xf972, + + 0xf97e, 0xf98a, 0xf997, 0xf9a3, 0xf9af, 0xf9bb, 0xf9c8, 0xf9d4, + 0xf9e1, 0xf9ed, 0xf9f9, 0xfa06, 0xfa12, 0xfa1f, 0xfa2b, 0xfa38, + 0xfa44, 0xfa51, 0xfa5d, 0xfa6a, 0xfa76, 0xfa83, 0xfa8f, 0xfa9c, + 0xfaa9, 0xfab5, 0xfac2, 0xfacf, 0xfadb, 0xfae8, 0xfaf5, 0xfb02, + 0xfb0e, 0xfb1b, 0xfb28, 0xfb35, 0xfb41, 0xfb4e, 0xfb5b, 0xfb68, + 0xfb75, 0xfb82, 0xfb8e, 0xfb9b, 0xfba8, 0xfbb5, 0xfbc2, 0xfbcf, + 0xfbdc, 0xfbe9, 0xfbf6, 0xfc03, 0xfc10, 0xfc1d, 0xfc2a, 0xfc37, + 0xfc44, 0xfc51, 0xfc5e, 0xfc6b, 0xfc78, 0xfc85, 0xfc92, 0xfc9f, + + 0xfcac, 0xfcba, 0xfcc7, 0xfcd4, 0xfce1, 0xfcee, 0xfcfb, 0xfd08, + 0xfd16, 0xfd23, 0xfd30, 0xfd3d, 0xfd4a, 0xfd58, 0xfd65, 0xfd72, + 0xfd7f, 0xfd8c, 0xfd9a, 0xfda7, 0xfdb4, 0xfdc2, 0xfdcf, 0xfddc, + 0xfde9, 0xfdf7, 0xfe04, 0xfe11, 0xfe1f, 0xfe2c, 0xfe39, 0xfe47, + 0xfe54, 0xfe61, 0xfe6f, 0xfe7c, 0xfe89, 0xfe97, 0xfea4, 0xfeb1, + 0xfebf, 0xfecc, 0xfed9, 0xfee7, 0xfef4, 0xff01, 0xff0f, 0xff1c, + 0xff2a, 0xff37, 0xff44, 0xff52, 0xff5f, 0xff6d, 0xff7a, 0xff87, + 0xff95, 0xffa2, 0xffb0, 0xffbd, 0xffca, 0xffd8, 0xffe5, 0xfff3, + + 0x0000, 0x000d, 0x001b, 0x0028, 0x0036, 0x0043, 0x0050, 0x005e, + 0x006b, 0x0079, 0x0086, 0x0093, 0x00a1, 0x00ae, 0x00bc, 0x00c9, + 0x00d6, 0x00e4, 0x00f1, 0x00ff, 0x010c, 0x0119, 0x0127, 0x0134, + 0x0141, 0x014f, 0x015c, 0x0169, 0x0177, 0x0184, 0x0191, 0x019f, + 0x01ac, 0x01b9, 0x01c7, 0x01d4, 0x01e1, 0x01ef, 0x01fc, 0x0209, + 0x0217, 0x0224, 0x0231, 0x023e, 0x024c, 0x0259, 0x0266, 0x0274, + 0x0281, 0x028e, 0x029b, 0x02a8, 0x02b6, 0x02c3, 0x02d0, 0x02dd, + 0x02ea, 0x02f8, 0x0305, 0x0312, 0x031f, 0x032c, 0x0339, 0x0346, + + 0x0354, 0x0361, 0x036e, 0x037b, 0x0388, 0x0395, 0x03a2, 0x03af, + 0x03bc, 0x03c9, 0x03d6, 0x03e3, 0x03f0, 0x03fd, 0x040a, 0x0417, + 0x0424, 0x0431, 0x043e, 0x044b, 0x0458, 0x0465, 0x0472, 0x047e, + 0x048b, 0x0498, 0x04a5, 0x04b2, 0x04bf, 0x04cb, 0x04d8, 0x04e5, + 0x04f2, 0x04fe, 0x050b, 0x0518, 0x0525, 0x0531, 0x053e, 0x054b, + 0x0557, 0x0564, 0x0571, 0x057d, 0x058a, 0x0596, 0x05a3, 0x05af, + 0x05bc, 0x05c8, 0x05d5, 0x05e1, 0x05ee, 0x05fa, 0x0607, 0x0613, + 0x061f, 0x062c, 0x0638, 0x0645, 0x0651, 0x065d, 0x0669, 0x0676, + + 0x0682, 0x068e, 0x069a, 0x06a7, 0x06b3, 0x06bf, 0x06cb, 0x06d7, + 0x06e3, 0x06ef, 0x06fc, 0x0708, 0x0714, 0x0720, 0x072c, 0x0738, + 0x0744, 0x074f, 0x075b, 0x0767, 0x0773, 0x077f, 0x078b, 0x0797, + 0x07a2, 0x07ae, 0x07ba, 0x07c6, 0x07d1, 0x07dd, 0x07e9, 0x07f4, + 0x0800, 0x080c, 0x0817, 0x0823, 0x082e, 0x083a, 0x0845, 0x0851, + 0x085c, 0x0868, 0x0873, 0x087e, 0x088a, 0x0895, 0x08a0, 0x08ac, + 0x08b7, 0x08c2, 0x08cd, 0x08d8, 0x08e4, 0x08ef, 0x08fa, 0x0905, + 0x0910, 0x091b, 0x0926, 0x0931, 0x093c, 0x0947, 0x0952, 0x095d, + + 0x0968, 0x0972, 0x097d, 0x0988, 0x0993, 0x099d, 0x09a8, 0x09b3, + 0x09bd, 0x09c8, 0x09d3, 0x09dd, 0x09e8, 0x09f2, 0x09fd, 0x0a07, + 0x0a12, 0x0a1c, 0x0a26, 0x0a31, 0x0a3b, 0x0a45, 0x0a50, 0x0a5a, + 0x0a64, 0x0a6e, 0x0a78, 0x0a83, 0x0a8d, 0x0a97, 0x0aa1, 0x0aab, + 0x0ab5, 0x0abf, 0x0ac9, 0x0ad3, 0x0adc, 0x0ae6, 0x0af0, 0x0afa, + 0x0b03, 0x0b0d, 0x0b17, 0x0b21, 0x0b2a, 0x0b34, 0x0b3d, 0x0b47, + 0x0b50, 0x0b5a, 0x0b63, 0x0b6d, 0x0b76, 0x0b7f, 0x0b89, 0x0b92, + 0x0b9b, 0x0ba4, 0x0bae, 0x0bb7, 0x0bc0, 0x0bc9, 0x0bd2, 0x0bdb, + + 0x0be4, 0x0bed, 0x0bf6, 0x0bff, 0x0c08, 0x0c10, 0x0c19, 0x0c22, + 0x0c2b, 0x0c33, 0x0c3c, 0x0c45, 0x0c4d, 0x0c56, 0x0c5e, 0x0c67, + 0x0c6f, 0x0c78, 0x0c80, 0x0c88, 0x0c91, 0x0c99, 0x0ca1, 0x0ca9, + 0x0cb2, 0x0cba, 0x0cc2, 0x0cca, 0x0cd2, 0x0cda, 0x0ce2, 0x0cea, + 0x0cf2, 0x0cfa, 0x0d01, 0x0d09, 0x0d11, 0x0d19, 0x0d20, 0x0d28, + 0x0d30, 0x0d37, 0x0d3f, 0x0d46, 0x0d4e, 0x0d55, 0x0d5d, 0x0d64, + 0x0d6b, 0x0d72, 0x0d7a, 0x0d81, 0x0d88, 0x0d8f, 0x0d96, 0x0d9d, + 0x0da4, 0x0dab, 0x0db2, 0x0db9, 0x0dc0, 0x0dc7, 0x0dce, 0x0dd5, + + 0x0ddb, 0x0de2, 0x0de9, 0x0def, 0x0df6, 0x0dfc, 0x0e03, 0x0e09, + 0x0e10, 0x0e16, 0x0e1c, 0x0e23, 0x0e29, 0x0e2f, 0x0e35, 0x0e3b, + 0x0e42, 0x0e48, 0x0e4e, 0x0e54, 0x0e5a, 0x0e5f, 0x0e65, 0x0e6b, + 0x0e71, 0x0e77, 0x0e7c, 0x0e82, 0x0e88, 0x0e8d, 0x0e93, 0x0e98, + 0x0e9e, 0x0ea3, 0x0ea9, 0x0eae, 0x0eb3, 0x0eb9, 0x0ebe, 0x0ec3, + 0x0ec8, 0x0ecd, 0x0ed2, 0x0ed7, 0x0edc, 0x0ee1, 0x0ee6, 0x0eeb, + 0x0ef0, 0x0ef5, 0x0ef9, 0x0efe, 0x0f03, 0x0f07, 0x0f0c, 0x0f11, + 0x0f15, 0x0f1a, 0x0f1e, 0x0f22, 0x0f27, 0x0f2b, 0x0f2f, 0x0f33, + + 0x0f38, 0x0f3c, 0x0f40, 0x0f44, 0x0f48, 0x0f4c, 0x0f50, 0x0f53, + 0x0f57, 0x0f5b, 0x0f5f, 0x0f63, 0x0f66, 0x0f6a, 0x0f6d, 0x0f71, + 0x0f74, 0x0f78, 0x0f7b, 0x0f7f, 0x0f82, 0x0f85, 0x0f88, 0x0f8c, + 0x0f8f, 0x0f92, 0x0f95, 0x0f98, 0x0f9b, 0x0f9e, 0x0fa1, 0x0fa4, + 0x0fa6, 0x0fa9, 0x0fac, 0x0faf, 0x0fb1, 0x0fb4, 0x0fb6, 0x0fb9, + 0x0fbb, 0x0fbe, 0x0fc0, 0x0fc3, 0x0fc5, 0x0fc7, 0x0fc9, 0x0fcb, + 0x0fce, 0x0fd0, 0x0fd2, 0x0fd4, 0x0fd6, 0x0fd8, 0x0fd9, 0x0fdb, + 0x0fdd, 0x0fdf, 0x0fe0, 0x0fe2, 0x0fe4, 0x0fe5, 0x0fe7, 0x0fe8, + + 0x0fea, 0x0feb, 0x0fec, 0x0fee, 0x0fef, 0x0ff0, 0x0ff1, 0x0ff2, + 0x0ff3, 0x0ff4, 0x0ff5, 0x0ff6, 0x0ff7, 0x0ff8, 0x0ff9, 0x0ffa, + 0x0ffa, 0x0ffb, 0x0ffc, 0x0ffc, 0x0ffd, 0x0ffd, 0x0ffe, 0x0ffe, + 0x0fff, 0x0fff, 0x0fff, 0x0fff, 0x1000, 0x1000, 0x1000, 0x1000, + 0x1000 +}; + +/* This is CSC = 1/SIN lookup table. NOTES: + * 1. The index ranges in value from 0 to 3*PI/2-1. + * 2. The value obtain from the table has 12 bits of fraction. + * 3. This value is used to calculate distance from: + * + * distance = distanceY * g_csc_table[angle] + * 4. 1/SIN(0) and 1/SIN(PI) are not defined. The value ZERO is place + * in the table -- Zero resulting range is a cue to take some other + * kind of action! + * 5. 1/COS(x) = 1/SIN(x + HALFPI) + */ + +const int32_t g_csc_table[TWOPI + HALFPI + 1] = +{ + 0x00000000, 0x00131940, 0x00098ca3, 0x00065dc6, 0x0004c658, + 0x0003d1e4, 0x00032eed, 0x0002ba86, 0x0002633a, 0x00021f54, + 0x0001e903, 0x0001bc93, 0x0001978b, 0x00017835, 0x00015d5b, + 0x00014615, 0x000131b8, 0x00011fc0, 0x00010fc8, 0x0001017f, + 0x0000f4a3, 0x0000e901, 0x0000de6e, 0x0000d4c7, 0x0000cbee, + 0x0000c3ca, 0x0000bc46, 0x0000b552, 0x0000aedc, 0x0000a8d9, + 0x0000a33d, 0x00009dfd, 0x00009912, 0x00009472, 0x00009019, + 0x00008c00, 0x00008820, 0x00008477, 0x000080ff, 0x00007db5, + + 0x00007a95, 0x0000779c, 0x000074c7, 0x00007214, 0x00006f81, + 0x00006d0b, 0x00006ab1, 0x00006870, 0x00006647, 0x00006436, + 0x00006239, 0x00006050, 0x00005e7b, 0x00005cb7, 0x00005b04, + 0x00005961, 0x000057cc, 0x00005647, 0x000054ce, 0x00005363, + 0x00005203, 0x000050b0, 0x00004f67, 0x00004e29, 0x00004cf5, + 0x00004bca, 0x00004aa9, 0x00004990, 0x0000487f, 0x00004777, + 0x00004676, 0x0000457c, 0x0000448a, 0x0000439e, 0x000042b9, + 0x000041d9, 0x00004100, 0x0000402c, 0x00003f5e, 0x00003e95, + + 0x00003dd2, 0x00003d13, 0x00003c59, 0x00003ba3, 0x00003af2, + 0x00003a45, 0x0000399c, 0x000038f7, 0x00003856, 0x000037b8, + 0x0000371e, 0x00003688, 0x000035f5, 0x00003565, 0x000034d8, + 0x0000344e, 0x000033c7, 0x00003343, 0x000032c1, 0x00003243, + 0x000031c7, 0x0000314d, 0x000030d6, 0x00003061, 0x00002fef, + 0x00002f7e, 0x00002f10, 0x00002ea4, 0x00002e3a, 0x00002dd2, + 0x00002d6c, 0x00002d08, 0x00002ca6, 0x00002c45, 0x00002be6, + 0x00002b89, 0x00002b2e, 0x00002ad4, 0x00002a7b, 0x00002a25, + + 0x000029cf, 0x0000297c, 0x00002929, 0x000028d8, 0x00002888, + 0x0000283a, 0x000027ed, 0x000027a1, 0x00002756, 0x0000270d, + 0x000026c5, 0x0000267e, 0x00002638, 0x000025f3, 0x000025af, + 0x0000256c, 0x0000252a, 0x000024e9, 0x000024aa, 0x0000246b, + 0x0000242d, 0x000023f0, 0x000023b4, 0x00002379, 0x0000233e, + 0x00002305, 0x000022cc, 0x00002294, 0x0000225d, 0x00002227, + 0x000021f1, 0x000021bc, 0x00002188, 0x00002155, 0x00002122, + 0x000020f0, 0x000020bf, 0x0000208e, 0x0000205e, 0x0000202f, + + 0x00002000, 0x00001fd2, 0x00001fa4, 0x00001f77, 0x00001f4b, + 0x00001f1f, 0x00001ef4, 0x00001ec9, 0x00001e9f, 0x00001e76, + 0x00001e4d, 0x00001e24, 0x00001dfc, 0x00001dd4, 0x00001dad, + 0x00001d87, 0x00001d61, 0x00001d3b, 0x00001d16, 0x00001cf1, + 0x00001ccd, 0x00001ca9, 0x00001c85, 0x00001c62, 0x00001c40, + 0x00001c1d, 0x00001bfb, 0x00001bda, 0x00001bb9, 0x00001b98, + 0x00001b78, 0x00001b58, 0x00001b39, 0x00001b19, 0x00001afa, + 0x00001adc, 0x00001abe, 0x00001aa0, 0x00001a82, 0x00001a65, + + 0x00001a48, 0x00001a2c, 0x00001a10, 0x000019f4, 0x000019d8, + 0x000019bd, 0x000019a2, 0x00001987, 0x0000196d, 0x00001952, + 0x00001939, 0x0000191f, 0x00001906, 0x000018ed, 0x000018d4, + 0x000018bb, 0x000018a3, 0x0000188b, 0x00001873, 0x0000185c, + 0x00001844, 0x0000182d, 0x00001816, 0x00001800, 0x000017e9, + 0x000017d3, 0x000017bd, 0x000017a8, 0x00001792, 0x0000177d, + 0x00001768, 0x00001753, 0x0000173e, 0x0000172a, 0x00001716, + 0x00001702, 0x000016ee, 0x000016da, 0x000016c7, 0x000016b4, + + 0x000016a1, 0x0000168e, 0x0000167b, 0x00001669, 0x00001656, + 0x00001644, 0x00001632, 0x00001620, 0x0000160f, 0x000015fd, + 0x000015ec, 0x000015db, 0x000015ca, 0x000015b9, 0x000015a9, + 0x00001598, 0x00001588, 0x00001578, 0x00001568, 0x00001558, + 0x00001548, 0x00001538, 0x00001529, 0x0000151a, 0x0000150b, + 0x000014fc, 0x000014ed, 0x000014de, 0x000014cf, 0x000014c1, + 0x000014b3, 0x000014a5, 0x00001497, 0x00001489, 0x0000147b, + 0x0000146d, 0x00001460, 0x00001452, 0x00001445, 0x00001438, + + 0x0000142b, 0x0000141e, 0x00001411, 0x00001405, 0x000013f8, + 0x000013ec, 0x000013df, 0x000013d3, 0x000013c7, 0x000013bb, + 0x000013af, 0x000013a3, 0x00001398, 0x0000138c, 0x00001381, + 0x00001375, 0x0000136a, 0x0000135f, 0x00001354, 0x00001349, + 0x0000133e, 0x00001333, 0x00001329, 0x0000131e, 0x00001314, + 0x0000130a, 0x000012ff, 0x000012f5, 0x000012eb, 0x000012e1, + 0x000012d7, 0x000012ce, 0x000012c4, 0x000012ba, 0x000012b1, + 0x000012a7, 0x0000129e, 0x00001295, 0x0000128c, 0x00001283, + + 0x0000127a, 0x00001271, 0x00001268, 0x0000125f, 0x00001257, + 0x0000124e, 0x00001246, 0x0000123d, 0x00001235, 0x0000122d, + 0x00001224, 0x0000121c, 0x00001214, 0x0000120c, 0x00001205, + 0x000011fd, 0x000011f5, 0x000011ed, 0x000011e6, 0x000011de, + 0x000011d7, 0x000011d0, 0x000011c8, 0x000011c1, 0x000011ba, + 0x000011b3, 0x000011ac, 0x000011a5, 0x0000119e, 0x00001198, + 0x00001191, 0x0000118a, 0x00001184, 0x0000117d, 0x00001177, + 0x00001170, 0x0000116a, 0x00001164, 0x0000115e, 0x00001158, + + 0x00001151, 0x0000114c, 0x00001146, 0x00001140, 0x0000113a, + 0x00001134, 0x0000112f, 0x00001129, 0x00001123, 0x0000111e, + 0x00001119, 0x00001113, 0x0000110e, 0x00001109, 0x00001103, + 0x000010fe, 0x000010f9, 0x000010f4, 0x000010ef, 0x000010ea, + 0x000010e6, 0x000010e1, 0x000010dc, 0x000010d7, 0x000010d3, + 0x000010ce, 0x000010ca, 0x000010c5, 0x000010c1, 0x000010bd, + 0x000010b8, 0x000010b4, 0x000010b0, 0x000010ac, 0x000010a8, + 0x000010a4, 0x000010a0, 0x0000109c, 0x00001098, 0x00001094, + + 0x00001090, 0x0000108d, 0x00001089, 0x00001086, 0x00001082, + 0x0000107f, 0x0000107b, 0x00001078, 0x00001074, 0x00001071, + 0x0000106e, 0x0000106b, 0x00001068, 0x00001064, 0x00001061, + 0x0000105e, 0x0000105c, 0x00001059, 0x00001056, 0x00001053, + 0x00001050, 0x0000104e, 0x0000104b, 0x00001048, 0x00001046, + 0x00001043, 0x00001041, 0x0000103e, 0x0000103c, 0x0000103a, + 0x00001037, 0x00001035, 0x00001033, 0x00001031, 0x0000102f, + 0x0000102d, 0x0000102b, 0x00001029, 0x00001027, 0x00001025, + + 0x00001023, 0x00001022, 0x00001020, 0x0000101e, 0x0000101d, + 0x0000101b, 0x00001019, 0x00001018, 0x00001017, 0x00001015, + 0x00001014, 0x00001013, 0x00001011, 0x00001010, 0x0000100f, + 0x0000100e, 0x0000100d, 0x0000100c, 0x0000100b, 0x0000100a, + 0x00001009, 0x00001008, 0x00001007, 0x00001006, 0x00001006, + 0x00001005, 0x00001004, 0x00001004, 0x00001003, 0x00001003, + 0x00001002, 0x00001002, 0x00001001, 0x00001001, 0x00001001, + 0x00001001, 0x00001000, 0x00001000, 0x00001000, 0x00001000, + + 0x00001000, 0x00001000, 0x00001000, 0x00001000, 0x00001000, + 0x00001001, 0x00001001, 0x00001001, 0x00001001, 0x00001002, + 0x00001002, 0x00001003, 0x00001003, 0x00001004, 0x00001004, + 0x00001005, 0x00001006, 0x00001006, 0x00001007, 0x00001008, + 0x00001009, 0x0000100a, 0x0000100b, 0x0000100c, 0x0000100d, + 0x0000100e, 0x0000100f, 0x00001010, 0x00001011, 0x00001013, + 0x00001014, 0x00001015, 0x00001017, 0x00001018, 0x00001019, + 0x0000101b, 0x0000101d, 0x0000101e, 0x00001020, 0x00001022, + + 0x00001023, 0x00001025, 0x00001027, 0x00001029, 0x0000102b, + 0x0000102d, 0x0000102f, 0x00001031, 0x00001033, 0x00001035, + 0x00001037, 0x0000103a, 0x0000103c, 0x0000103e, 0x00001041, + 0x00001043, 0x00001046, 0x00001048, 0x0000104b, 0x0000104e, + 0x00001050, 0x00001053, 0x00001056, 0x00001059, 0x0000105c, + 0x0000105e, 0x00001061, 0x00001064, 0x00001068, 0x0000106b, + 0x0000106e, 0x00001071, 0x00001074, 0x00001078, 0x0000107b, + 0x0000107f, 0x00001082, 0x00001086, 0x00001089, 0x0000108d, + + 0x00001090, 0x00001094, 0x00001098, 0x0000109c, 0x000010a0, + 0x000010a4, 0x000010a8, 0x000010ac, 0x000010b0, 0x000010b4, + 0x000010b8, 0x000010bd, 0x000010c1, 0x000010c5, 0x000010ca, + 0x000010ce, 0x000010d3, 0x000010d7, 0x000010dc, 0x000010e1, + 0x000010e6, 0x000010ea, 0x000010ef, 0x000010f4, 0x000010f9, + 0x000010fe, 0x00001103, 0x00001109, 0x0000110e, 0x00001113, + 0x00001119, 0x0000111e, 0x00001123, 0x00001129, 0x0000112f, + 0x00001134, 0x0000113a, 0x00001140, 0x00001146, 0x0000114c, + + 0x00001151, 0x00001158, 0x0000115e, 0x00001164, 0x0000116a, + 0x00001170, 0x00001177, 0x0000117d, 0x00001184, 0x0000118a, + 0x00001191, 0x00001198, 0x0000119e, 0x000011a5, 0x000011ac, + 0x000011b3, 0x000011ba, 0x000011c1, 0x000011c8, 0x000011d0, + 0x000011d7, 0x000011de, 0x000011e6, 0x000011ed, 0x000011f5, + 0x000011fd, 0x00001205, 0x0000120c, 0x00001214, 0x0000121c, + 0x00001224, 0x0000122d, 0x00001235, 0x0000123d, 0x00001246, + 0x0000124e, 0x00001257, 0x0000125f, 0x00001268, 0x00001271, + + 0x0000127a, 0x00001283, 0x0000128c, 0x00001295, 0x0000129e, + 0x000012a7, 0x000012b1, 0x000012ba, 0x000012c4, 0x000012ce, + 0x000012d7, 0x000012e1, 0x000012eb, 0x000012f5, 0x000012ff, + 0x0000130a, 0x00001314, 0x0000131e, 0x00001329, 0x00001333, + 0x0000133e, 0x00001349, 0x00001354, 0x0000135f, 0x0000136a, + 0x00001375, 0x00001381, 0x0000138c, 0x00001398, 0x000013a3, + 0x000013af, 0x000013bb, 0x000013c7, 0x000013d3, 0x000013df, + 0x000013ec, 0x000013f8, 0x00001405, 0x00001411, 0x0000141e, + + 0x0000142b, 0x00001438, 0x00001445, 0x00001452, 0x00001460, + 0x0000146d, 0x0000147b, 0x00001489, 0x00001497, 0x000014a5, + 0x000014b3, 0x000014c1, 0x000014cf, 0x000014de, 0x000014ed, + 0x000014fc, 0x0000150b, 0x0000151a, 0x00001529, 0x00001538, + 0x00001548, 0x00001558, 0x00001568, 0x00001578, 0x00001588, + 0x00001598, 0x000015a9, 0x000015b9, 0x000015ca, 0x000015db, + 0x000015ec, 0x000015fd, 0x0000160f, 0x00001620, 0x00001632, + 0x00001644, 0x00001656, 0x00001669, 0x0000167b, 0x0000168e, + + 0x000016a1, 0x000016b4, 0x000016c7, 0x000016da, 0x000016ee, + 0x00001702, 0x00001716, 0x0000172a, 0x0000173e, 0x00001753, + 0x00001768, 0x0000177d, 0x00001792, 0x000017a8, 0x000017bd, + 0x000017d3, 0x000017e9, 0x00001800, 0x00001816, 0x0000182d, + 0x00001844, 0x0000185c, 0x00001873, 0x0000188b, 0x000018a3, + 0x000018bb, 0x000018d4, 0x000018ed, 0x00001906, 0x0000191f, + 0x00001939, 0x00001952, 0x0000196d, 0x00001987, 0x000019a2, + 0x000019bd, 0x000019d8, 0x000019f4, 0x00001a10, 0x00001a2c, + + 0x00001a48, 0x00001a65, 0x00001a82, 0x00001aa0, 0x00001abe, + 0x00001adc, 0x00001afa, 0x00001b19, 0x00001b39, 0x00001b58, + 0x00001b78, 0x00001b98, 0x00001bb9, 0x00001bda, 0x00001bfb, + 0x00001c1d, 0x00001c40, 0x00001c62, 0x00001c85, 0x00001ca9, + 0x00001ccd, 0x00001cf1, 0x00001d16, 0x00001d3b, 0x00001d61, + 0x00001d87, 0x00001dad, 0x00001dd4, 0x00001dfc, 0x00001e24, + 0x00001e4d, 0x00001e76, 0x00001e9f, 0x00001ec9, 0x00001ef4, + 0x00001f1f, 0x00001f4b, 0x00001f77, 0x00001fa4, 0x00001fd2, + + 0x00002000, 0x0000202f, 0x0000205e, 0x0000208e, 0x000020bf, + 0x000020f0, 0x00002122, 0x00002155, 0x00002188, 0x000021bc, + 0x000021f1, 0x00002227, 0x0000225d, 0x00002294, 0x000022cc, + 0x00002305, 0x0000233e, 0x00002379, 0x000023b4, 0x000023f0, + 0x0000242d, 0x0000246b, 0x000024aa, 0x000024e9, 0x0000252a, + 0x0000256c, 0x000025af, 0x000025f3, 0x00002638, 0x0000267e, + 0x000026c5, 0x0000270d, 0x00002756, 0x000027a1, 0x000027ed, + 0x0000283a, 0x00002888, 0x000028d8, 0x00002929, 0x0000297c, + + 0x000029cf, 0x00002a25, 0x00002a7b, 0x00002ad4, 0x00002b2e, + 0x00002b89, 0x00002be6, 0x00002c45, 0x00002ca6, 0x00002d08, + 0x00002d6c, 0x00002dd2, 0x00002e3a, 0x00002ea4, 0x00002f10, + 0x00002f7e, 0x00002fef, 0x00003061, 0x000030d6, 0x0000314d, + 0x000031c7, 0x00003243, 0x000032c1, 0x00003343, 0x000033c7, + 0x0000344e, 0x000034d8, 0x00003565, 0x000035f5, 0x00003688, + 0x0000371e, 0x000037b8, 0x00003856, 0x000038f7, 0x0000399c, + 0x00003a45, 0x00003af2, 0x00003ba3, 0x00003c59, 0x00003d13, + + 0x00003dd2, 0x00003e95, 0x00003f5e, 0x0000402c, 0x00004100, + 0x000041d9, 0x000042b9, 0x0000439e, 0x0000448a, 0x0000457c, + 0x00004676, 0x00004777, 0x0000487f, 0x00004990, 0x00004aa9, + 0x00004bca, 0x00004cf5, 0x00004e29, 0x00004f67, 0x000050b0, + 0x00005203, 0x00005363, 0x000054ce, 0x00005647, 0x000057cc, + 0x00005961, 0x00005b04, 0x00005cb7, 0x00005e7b, 0x00006050, + 0x00006239, 0x00006436, 0x00006647, 0x00006870, 0x00006ab1, + 0x00006d0b, 0x00006f81, 0x00007214, 0x000074c7, 0x0000779c, + + 0x00007a95, 0x00007db5, 0x000080ff, 0x00008477, 0x00008820, + 0x00008c00, 0x00009019, 0x00009472, 0x00009911, 0x00009dfd, + 0x0000a33d, 0x0000a8d9, 0x0000aedc, 0x0000b552, 0x0000bc46, + 0x0000c3ca, 0x0000cbee, 0x0000d4c7, 0x0000de6e, 0x0000e901, + 0x0000f4a3, 0x0001017f, 0x00010fc8, 0x00011fc0, 0x000131b7, + 0x00014614, 0x00015d5b, 0x00017836, 0x0001978a, 0x0001bc92, + 0x0001e903, 0x00021f54, 0x0002633a, 0x0002ba86, 0x00032eeb, + 0x0003d1e2, 0x0004c656, 0x00065dc4, 0x00098ca4, 0x00131951, + + 0x00000000, 0xffece696, 0xfff67356, 0xfff9a239, 0xfffb39a8, + 0xfffc2e1d, 0xfffcd112, 0xfffd4579, 0xfffd9cc6, 0xfffde0ac, + 0xfffe16fd, 0xfffe436e, 0xfffe6875, 0xfffe87ca, 0xfffea2a5, + 0xfffeb9eb, 0xfffece48, 0xfffee040, 0xfffef038, 0xfffefe81, + 0xffff0b5d, 0xffff16ff, 0xffff2192, 0xffff2b39, 0xffff3412, + 0xffff3c36, 0xffff43ba, 0xffff4aae, 0xffff5124, 0xffff5727, + 0xffff5cc3, 0xffff6203, 0xffff66ee, 0xffff6b8e, 0xffff6fe7, + 0xffff7400, 0xffff77df, 0xffff7b89, 0xffff7f01, 0xffff824b, + + 0xffff856b, 0xffff8864, 0xffff8b39, 0xffff8dec, 0xffff907f, + 0xffff92f5, 0xffff954f, 0xffff9790, 0xffff99b8, 0xffff9bca, + 0xffff9dc7, 0xffff9fb0, 0xffffa185, 0xffffa349, 0xffffa4fc, + 0xffffa69f, 0xffffa834, 0xffffa9b9, 0xffffab32, 0xffffac9d, + 0xffffadfd, 0xffffaf50, 0xffffb099, 0xffffb1d7, 0xffffb30b, + 0xffffb436, 0xffffb557, 0xffffb670, 0xffffb781, 0xffffb889, + 0xffffb98a, 0xffffba84, 0xffffbb76, 0xffffbc62, 0xffffbd47, + 0xffffbe27, 0xffffbf00, 0xffffbfd4, 0xffffc0a2, 0xffffc16b, + + 0xffffc22e, 0xffffc2ed, 0xffffc3a7, 0xffffc45d, 0xffffc50e, + 0xffffc5bb, 0xffffc664, 0xffffc709, 0xffffc7aa, 0xffffc848, + 0xffffc8e2, 0xffffc978, 0xffffca0b, 0xffffca9b, 0xffffcb28, + 0xffffcbb2, 0xffffcc39, 0xffffccbd, 0xffffcd3f, 0xffffcdbd, + 0xffffce39, 0xffffceb3, 0xffffcf2a, 0xffffcf9f, 0xffffd011, + 0xffffd082, 0xffffd0f0, 0xffffd15c, 0xffffd1c6, 0xffffd22e, + 0xffffd294, 0xffffd2f8, 0xffffd35a, 0xffffd3bb, 0xffffd41a, + 0xffffd477, 0xffffd4d2, 0xffffd52c, 0xffffd585, 0xffffd5db, + + 0xffffd631, 0xffffd684, 0xffffd6d7, 0xffffd728, 0xffffd778, + 0xffffd7c6, 0xffffd813, 0xffffd85f, 0xffffd8aa, 0xffffd8f3, + 0xffffd93b, 0xffffd982, 0xffffd9c8, 0xffffda0d, 0xffffda51, + 0xffffda94, 0xffffdad6, 0xffffdb17, 0xffffdb56, 0xffffdb95, + 0xffffdbd3, 0xffffdc10, 0xffffdc4c, 0xffffdc87, 0xffffdcc2, + 0xffffdcfb, 0xffffdd34, 0xffffdd6c, 0xffffdda3, 0xffffddd9, + 0xffffde0f, 0xffffde44, 0xffffde78, 0xffffdeab, 0xffffdede, + 0xffffdf10, 0xffffdf41, 0xffffdf72, 0xffffdfa2, 0xffffdfd1, + + 0xffffe000, 0xffffe02e, 0xffffe05c, 0xffffe089, 0xffffe0b5, + 0xffffe0e1, 0xffffe10c, 0xffffe137, 0xffffe161, 0xffffe18a, + 0xffffe1b3, 0xffffe1dc, 0xffffe204, 0xffffe22c, 0xffffe253, + 0xffffe279, 0xffffe29f, 0xffffe2c5, 0xffffe2ea, 0xffffe30f, + 0xffffe333, 0xffffe357, 0xffffe37b, 0xffffe39e, 0xffffe3c0, + 0xffffe3e3, 0xffffe405, 0xffffe426, 0xffffe447, 0xffffe468, + 0xffffe488, 0xffffe4a8, 0xffffe4c7, 0xffffe4e7, 0xffffe506, + 0xffffe524, 0xffffe542, 0xffffe560, 0xffffe57e, 0xffffe59b, + + 0xffffe5b8, 0xffffe5d4, 0xffffe5f0, 0xffffe60c, 0xffffe628, + 0xffffe643, 0xffffe65e, 0xffffe679, 0xffffe693, 0xffffe6ae, + 0xffffe6c7, 0xffffe6e1, 0xffffe6fa, 0xffffe713, 0xffffe72c, + 0xffffe745, 0xffffe75d, 0xffffe775, 0xffffe78d, 0xffffe7a4, + 0xffffe7bc, 0xffffe7d3, 0xffffe7ea, 0xffffe800, 0xffffe817, + 0xffffe82d, 0xffffe843, 0xffffe858, 0xffffe86e, 0xffffe883, + 0xffffe898, 0xffffe8ad, 0xffffe8c2, 0xffffe8d6, 0xffffe8ea, + 0xffffe8fe, 0xffffe912, 0xffffe926, 0xffffe939, 0xffffe94c, + + 0xffffe95f, 0xffffe972, 0xffffe985, 0xffffe997, 0xffffe9aa, + 0xffffe9bc, 0xffffe9ce, 0xffffe9e0, 0xffffe9f1, 0xffffea03, + 0xffffea14, 0xffffea25, 0xffffea36, 0xffffea47, 0xffffea57, + 0xffffea68, 0xffffea78, 0xffffea88, 0xffffea98, 0xffffeaa8, + 0xffffeab8, 0xffffeac8, 0xffffead7, 0xffffeae6, 0xffffeaf5, + 0xffffeb04, 0xffffeb13, 0xffffeb22, 0xffffeb31, 0xffffeb3f, + 0xffffeb4d, 0xffffeb5b, 0xffffeb69, 0xffffeb77, 0xffffeb85, + 0xffffeb93, 0xffffeba0, 0xffffebae, 0xffffebbb, 0xffffebc8, + + 0xffffebd5, 0xffffebe2, 0xffffebef, 0xffffebfb, 0xffffec08, + 0xffffec14, 0xffffec21, 0xffffec2d, 0xffffec39, 0xffffec45, + 0xffffec51, 0xffffec5d, 0xffffec68, 0xffffec74, 0xffffec7f, + 0xffffec8b, 0xffffec96, 0xffffeca1, 0xffffecac, 0xffffecb7, + 0xffffecc2, 0xffffeccd, 0xffffecd7, 0xffffece2, 0xffffecec, + 0xffffecf6, 0xffffed01, 0xffffed0b, 0xffffed15, 0xffffed1f, + 0xffffed29, 0xffffed32, 0xffffed3c, 0xffffed46, 0xffffed4f, + 0xffffed59, 0xffffed62, 0xffffed6b, 0xffffed74, 0xffffed7d, + + 0xffffed86, 0xffffed8f, 0xffffed98, 0xffffeda1, 0xffffeda9, + 0xffffedb2, 0xffffedba, 0xffffedc3, 0xffffedcb, 0xffffedd3, + 0xffffeddc, 0xffffede4, 0xffffedec, 0xffffedf4, 0xffffedfb, + 0xffffee03, 0xffffee0b, 0xffffee13, 0xffffee1a, 0xffffee22, + 0xffffee29, 0xffffee30, 0xffffee38, 0xffffee3f, 0xffffee46, + 0xffffee4d, 0xffffee54, 0xffffee5b, 0xffffee62, 0xffffee68, + 0xffffee6f, 0xffffee76, 0xffffee7c, 0xffffee83, 0xffffee89, + 0xffffee90, 0xffffee96, 0xffffee9c, 0xffffeea2, 0xffffeea8, + + 0xffffeeaf, 0xffffeeb4, 0xffffeeba, 0xffffeec0, 0xffffeec6, + 0xffffeecc, 0xffffeed1, 0xffffeed7, 0xffffeedd, 0xffffeee2, + 0xffffeee7, 0xffffeeed, 0xffffeef2, 0xffffeef7, 0xffffeefd, + 0xffffef02, 0xffffef07, 0xffffef0c, 0xffffef11, 0xffffef16, + 0xffffef1a, 0xffffef1f, 0xffffef24, 0xffffef29, 0xffffef2d, + 0xffffef32, 0xffffef36, 0xffffef3b, 0xffffef3f, 0xffffef43, + 0xffffef48, 0xffffef4c, 0xffffef50, 0xffffef54, 0xffffef58, + 0xffffef5c, 0xffffef60, 0xffffef64, 0xffffef68, 0xffffef6c, + + 0xffffef70, 0xffffef73, 0xffffef77, 0xffffef7a, 0xffffef7e, + 0xffffef81, 0xffffef85, 0xffffef88, 0xffffef8c, 0xffffef8f, + 0xffffef92, 0xffffef95, 0xffffef98, 0xffffef9c, 0xffffef9f, + 0xffffefa2, 0xffffefa4, 0xffffefa7, 0xffffefaa, 0xffffefad, + 0xffffefb0, 0xffffefb2, 0xffffefb5, 0xffffefb8, 0xffffefba, + 0xffffefbd, 0xffffefbf, 0xffffefc2, 0xffffefc4, 0xffffefc6, + 0xffffefc9, 0xffffefcb, 0xffffefcd, 0xffffefcf, 0xffffefd1, + 0xffffefd3, 0xffffefd5, 0xffffefd7, 0xffffefd9, 0xffffefdb, + + 0xffffefdd, 0xffffefde, 0xffffefe0, 0xffffefe2, 0xffffefe3, + 0xffffefe5, 0xffffefe7, 0xffffefe8, 0xffffefe9, 0xffffefeb, + 0xffffefec, 0xffffefed, 0xffffefef, 0xffffeff0, 0xffffeff1, + 0xffffeff2, 0xffffeff3, 0xffffeff4, 0xffffeff5, 0xffffeff6, + 0xffffeff7, 0xffffeff8, 0xffffeff9, 0xffffeffa, 0xffffeffa, + 0xffffeffb, 0xffffeffc, 0xffffeffc, 0xffffeffd, 0xffffeffd, + 0xffffeffe, 0xffffeffe, 0xffffefff, 0xffffefff, 0xffffefff, + 0xffffefff, 0xfffff000, 0xfffff000, 0xfffff000, 0xfffff000, + + 0xfffff000, 0xfffff000, 0xfffff000, 0xfffff000, 0xfffff000, + 0xffffefff, 0xffffefff, 0xffffefff, 0xffffefff, 0xffffeffe, + 0xffffeffe, 0xffffeffd, 0xffffeffd, 0xffffeffc, 0xffffeffc, + 0xffffeffb, 0xffffeffa, 0xffffeffa, 0xffffeff9, 0xffffeff8, + 0xffffeff7, 0xffffeff6, 0xffffeff5, 0xffffeff4, 0xffffeff3, + 0xffffeff2, 0xffffeff1, 0xffffeff0, 0xffffefef, 0xffffefed, + 0xffffefec, 0xffffefeb, 0xffffefe9, 0xffffefe8, 0xffffefe7, + 0xffffefe5, 0xffffefe3, 0xffffefe2, 0xffffefe0, 0xffffefde, + + 0xffffefdd, 0xffffefdb, 0xffffefd9, 0xffffefd7, 0xffffefd5, + 0xffffefd3, 0xffffefd1, 0xffffefcf, 0xffffefcd, 0xffffefcb, + 0xffffefc9, 0xffffefc6, 0xffffefc4, 0xffffefc2, 0xffffefbf, + 0xffffefbd, 0xffffefba, 0xffffefb8, 0xffffefb5, 0xffffefb2, + 0xffffefb0, 0xffffefad, 0xffffefaa, 0xffffefa7, 0xffffefa4, + 0xffffefa2, 0xffffef9f, 0xffffef9c, 0xffffef98, 0xffffef95, + 0xffffef92, 0xffffef8f, 0xffffef8c, 0xffffef88, 0xffffef85, + 0xffffef81, 0xffffef7e, 0xffffef7a, 0xffffef77, 0xffffef73, + + 0xffffef70, 0xffffef6c, 0xffffef68, 0xffffef64, 0xffffef60, + 0xffffef5c, 0xffffef58, 0xffffef54, 0xffffef50, 0xffffef4c, + 0xffffef48, 0xffffef43, 0xffffef3f, 0xffffef3b, 0xffffef36, + 0xffffef32, 0xffffef2d, 0xffffef29, 0xffffef24, 0xffffef1f, + 0xffffef1a, 0xffffef16, 0xffffef11, 0xffffef0c, 0xffffef07, + 0xffffef02, 0xffffeefd, 0xffffeef7, 0xffffeef2, 0xffffeeed, + 0xffffeee7, 0xffffeee2, 0xffffeedd, 0xffffeed7, 0xffffeed1, + 0xffffeecc, 0xffffeec6, 0xffffeec0, 0xffffeeba, 0xffffeeb4, + + 0xffffeeaf, 0xffffeea8, 0xffffeea2, 0xffffee9c, 0xffffee96, + 0xffffee90, 0xffffee89, 0xffffee83, 0xffffee7c, 0xffffee76, + 0xffffee6f, 0xffffee68, 0xffffee62, 0xffffee5b, 0xffffee54, + 0xffffee4d, 0xffffee46, 0xffffee3f, 0xffffee38, 0xffffee30, + 0xffffee29, 0xffffee22, 0xffffee1a, 0xffffee13, 0xffffee0b, + 0xffffee03, 0xffffedfb, 0xffffedf4, 0xffffedec, 0xffffede4, + 0xffffeddc, 0xffffedd3, 0xffffedcb, 0xffffedc3, 0xffffedba, + 0xffffedb2, 0xffffeda9, 0xffffeda1, 0xffffed98, 0xffffed8f, + + 0xffffed86, 0xffffed7d, 0xffffed74, 0xffffed6b, 0xffffed62, + 0xffffed59, 0xffffed4f, 0xffffed46, 0xffffed3c, 0xffffed32, + 0xffffed29, 0xffffed1f, 0xffffed15, 0xffffed0b, 0xffffed01, + 0xffffecf6, 0xffffecec, 0xffffece2, 0xffffecd7, 0xffffeccd, + 0xffffecc2, 0xffffecb7, 0xffffecac, 0xffffeca1, 0xffffec96, + 0xffffec8b, 0xffffec7f, 0xffffec74, 0xffffec68, 0xffffec5d, + 0xffffec51, 0xffffec45, 0xffffec39, 0xffffec2d, 0xffffec21, + 0xffffec14, 0xffffec08, 0xffffebfb, 0xffffebef, 0xffffebe2, + + 0xffffebd5, 0xffffebc8, 0xffffebbb, 0xffffebae, 0xffffeba0, + 0xffffeb93, 0xffffeb85, 0xffffeb77, 0xffffeb69, 0xffffeb5b, + 0xffffeb4d, 0xffffeb3f, 0xffffeb31, 0xffffeb22, 0xffffeb13, + 0xffffeb04, 0xffffeaf5, 0xffffeae6, 0xffffead7, 0xffffeac8, + 0xffffeab8, 0xffffeaa8, 0xffffea98, 0xffffea88, 0xffffea78, + 0xffffea68, 0xffffea57, 0xffffea47, 0xffffea36, 0xffffea25, + 0xffffea14, 0xffffea03, 0xffffe9f1, 0xffffe9e0, 0xffffe9ce, + 0xffffe9bc, 0xffffe9aa, 0xffffe997, 0xffffe985, 0xffffe972, + + 0xffffe95f, 0xffffe94c, 0xffffe939, 0xffffe926, 0xffffe912, + 0xffffe8fe, 0xffffe8ea, 0xffffe8d6, 0xffffe8c2, 0xffffe8ad, + 0xffffe898, 0xffffe883, 0xffffe86e, 0xffffe858, 0xffffe843, + 0xffffe82d, 0xffffe817, 0xffffe800, 0xffffe7ea, 0xffffe7d3, + 0xffffe7bc, 0xffffe7a4, 0xffffe78d, 0xffffe775, 0xffffe75d, + 0xffffe745, 0xffffe72c, 0xffffe713, 0xffffe6fa, 0xffffe6e1, + 0xffffe6c7, 0xffffe6ae, 0xffffe693, 0xffffe679, 0xffffe65e, + 0xffffe643, 0xffffe628, 0xffffe60c, 0xffffe5f0, 0xffffe5d4, + + 0xffffe5b8, 0xffffe59b, 0xffffe57e, 0xffffe560, 0xffffe542, + 0xffffe524, 0xffffe506, 0xffffe4e7, 0xffffe4c7, 0xffffe4a8, + 0xffffe488, 0xffffe468, 0xffffe447, 0xffffe426, 0xffffe405, + 0xffffe3e3, 0xffffe3c0, 0xffffe39e, 0xffffe37b, 0xffffe357, + 0xffffe333, 0xffffe30f, 0xffffe2ea, 0xffffe2c5, 0xffffe29f, + 0xffffe279, 0xffffe253, 0xffffe22c, 0xffffe204, 0xffffe1dc, + 0xffffe1b3, 0xffffe18a, 0xffffe161, 0xffffe137, 0xffffe10c, + 0xffffe0e1, 0xffffe0b5, 0xffffe089, 0xffffe05c, 0xffffe02e, + + 0xffffe000, 0xffffdfd1, 0xffffdfa2, 0xffffdf72, 0xffffdf41, + 0xffffdf10, 0xffffdede, 0xffffdeab, 0xffffde78, 0xffffde44, + 0xffffde0f, 0xffffddd9, 0xffffdda3, 0xffffdd6c, 0xffffdd34, + 0xffffdcfb, 0xffffdcc2, 0xffffdc87, 0xffffdc4c, 0xffffdc10, + 0xffffdbd3, 0xffffdb95, 0xffffdb56, 0xffffdb17, 0xffffdad6, + 0xffffda94, 0xffffda51, 0xffffda0d, 0xffffd9c8, 0xffffd982, + 0xffffd93b, 0xffffd8f3, 0xffffd8aa, 0xffffd85f, 0xffffd813, + 0xffffd7c6, 0xffffd778, 0xffffd728, 0xffffd6d7, 0xffffd684, + + 0xffffd631, 0xffffd5db, 0xffffd585, 0xffffd52c, 0xffffd4d2, + 0xffffd477, 0xffffd41a, 0xffffd3bb, 0xffffd35a, 0xffffd2f8, + 0xffffd294, 0xffffd22e, 0xffffd1c6, 0xffffd15c, 0xffffd0f0, + 0xffffd082, 0xffffd011, 0xffffcf9f, 0xffffcf2a, 0xffffceb3, + 0xffffce39, 0xffffcdbd, 0xffffcd3f, 0xffffccbd, 0xffffcc39, + 0xffffcbb2, 0xffffcb28, 0xffffca9b, 0xffffca0b, 0xffffc978, + 0xffffc8e2, 0xffffc848, 0xffffc7aa, 0xffffc709, 0xffffc664, + 0xffffc5bb, 0xffffc50e, 0xffffc45d, 0xffffc3a7, 0xffffc2ed, + + 0xffffc22e, 0xffffc16b, 0xffffc0a2, 0xffffbfd4, 0xffffbf00, + 0xffffbe27, 0xffffbd47, 0xffffbc62, 0xffffbb76, 0xffffba84, + 0xffffb98a, 0xffffb889, 0xffffb781, 0xffffb670, 0xffffb557, + 0xffffb436, 0xffffb30b, 0xffffb1d7, 0xffffb099, 0xffffaf50, + 0xffffadfd, 0xffffac9d, 0xffffab32, 0xffffa9b9, 0xffffa834, + 0xffffa69f, 0xffffa4fc, 0xffffa349, 0xffffa185, 0xffff9fb0, + 0xffff9dc7, 0xffff9bca, 0xffff99b8, 0xffff9790, 0xffff954f, + 0xffff92f5, 0xffff907f, 0xffff8dec, 0xffff8b39, 0xffff8864, + + 0xffff856b, 0xffff824b, 0xffff7f01, 0xffff7b89, 0xffff77df, + 0xffff7400, 0xffff6fe7, 0xffff6b8e, 0xffff66ef, 0xffff6203, + 0xffff5cc3, 0xffff5727, 0xffff5124, 0xffff4aae, 0xffff43ba, + 0xffff3c36, 0xffff3413, 0xffff2b39, 0xffff2192, 0xffff16ff, + 0xffff0b5d, 0xfffefe82, 0xfffef038, 0xfffee040, 0xfffece48, + 0xfffeb9eb, 0xfffea2a5, 0xfffe87ca, 0xfffe6876, 0xfffe436e, + 0xfffe16fe, 0xfffde0ad, 0xfffd9cc7, 0xfffd457b, 0xfffcd114, + 0xfffc2e1c, 0xfffb39a7, 0xfff9a238, 0xfff67354, 0xffece743, + + 0x00000000, 0x001319a4, 0x00098cb8, 0x00065dcd, 0x0004c65c, + 0x0003d1e6, 0x00032eee, 0x0002ba86, 0x00026339, 0x00021f54, + 0x0001e902, 0x0001bc94, 0x0001978b, 0x00017836, 0x00015d5b, + 0x00014615, 0x000131b8, 0x00011fc0, 0x00010fc8, 0x0001017f, + 0x0000f4a3, 0x0000e901, 0x0000de6e, 0x0000d4c7, 0x0000cbee, + 0x0000c3ca, 0x0000bc46, 0x0000b552, 0x0000aedc, 0x0000a8d9, + 0x0000a33d, 0x00009dfd, 0x00009911, 0x00009472, 0x00009019, + 0x00008c00, 0x00008821, 0x00008477, 0x000080ff, 0x00007db5, + + 0x00007a95, 0x0000779c, 0x000074c7, 0x00007214, 0x00006f81, + 0x00006d0b, 0x00006ab1, 0x00006870, 0x00006648, 0x00006436, + 0x00006239, 0x00006050, 0x00005e7b, 0x00005cb7, 0x00005b04, + 0x00005961, 0x000057cc, 0x00005647, 0x000054ce, 0x00005363, + 0x00005203, 0x000050b0, 0x00004f67, 0x00004e29, 0x00004cf5, + 0x00004bca, 0x00004aa9, 0x00004990, 0x0000487f, 0x00004777, + 0x00004676, 0x0000457c, 0x0000448a, 0x0000439e, 0x000042b9, + 0x000041d9, 0x00004100, 0x0000402c, 0x00003f5e, 0x00003e95, + + 0x00003dd2, 0x00003d13, 0x00003c59, 0x00003ba3, 0x00003af2, + 0x00003a45, 0x0000399c, 0x000038f7, 0x00003856, 0x000037b8, + 0x0000371e, 0x00003688, 0x000035f5, 0x00003565, 0x000034d8, + 0x0000344e, 0x000033c7, 0x00003343, 0x000032c1, 0x00003243, + 0x000031c7, 0x0000314d, 0x000030d6, 0x00003061, 0x00002fef, + 0x00002f7e, 0x00002f10, 0x00002ea4, 0x00002e3a, 0x00002dd2, + 0x00002d6c, 0x00002d08, 0x00002ca6, 0x00002c45, 0x00002be6, + 0x00002b89, 0x00002b2e, 0x00002ad4, 0x00002a7b, 0x00002a25, + + 0x000029cf, 0x0000297c, 0x00002929, 0x000028d8, 0x00002888, + 0x0000283a, 0x000027ed, 0x000027a1, 0x00002756, 0x0000270d, + 0x000026c5, 0x0000267e, 0x00002638, 0x000025f3, 0x000025af, + 0x0000256c, 0x0000252a, 0x000024e9, 0x000024aa, 0x0000246b, + 0x0000242d, 0x000023f0, 0x000023b4, 0x00002379, 0x0000233e, + 0x00002305, 0x000022cc, 0x00002294, 0x0000225d, 0x00002227, + 0x000021f1, 0x000021bc, 0x00002188, 0x00002155, 0x00002122, + 0x000020f0, 0x000020bf, 0x0000208e, 0x0000205e, 0x0000202f, + + 0x00002000, 0x00001fd2, 0x00001fa4, 0x00001f77, 0x00001f4b, + 0x00001f1f, 0x00001ef4, 0x00001ec9, 0x00001e9f, 0x00001e76, + 0x00001e4d, 0x00001e24, 0x00001dfc, 0x00001dd4, 0x00001dad, + 0x00001d87, 0x00001d61, 0x00001d3b, 0x00001d16, 0x00001cf1, + 0x00001ccd, 0x00001ca9, 0x00001c85, 0x00001c62, 0x00001c40, + 0x00001c1d, 0x00001bfb, 0x00001bda, 0x00001bb9, 0x00001b98, + 0x00001b78, 0x00001b58, 0x00001b39, 0x00001b19, 0x00001afa, + 0x00001adc, 0x00001abe, 0x00001aa0, 0x00001a82, 0x00001a65, + + 0x00001a48, 0x00001a2c, 0x00001a10, 0x000019f4, 0x000019d8, + 0x000019bd, 0x000019a2, 0x00001987, 0x0000196d, 0x00001952, + 0x00001939, 0x0000191f, 0x00001906, 0x000018ed, 0x000018d4, + 0x000018bb, 0x000018a3, 0x0000188b, 0x00001873, 0x0000185c, + 0x00001844, 0x0000182d, 0x00001816, 0x00001800, 0x000017e9, + 0x000017d3, 0x000017bd, 0x000017a8, 0x00001792, 0x0000177d, + 0x00001768, 0x00001753, 0x0000173e, 0x0000172a, 0x00001716, + 0x00001702, 0x000016ee, 0x000016da, 0x000016c7, 0x000016b4, + + 0x000016a1, 0x0000168e, 0x0000167b, 0x00001669, 0x00001656, + 0x00001644, 0x00001632, 0x00001620, 0x0000160f, 0x000015fd, + 0x000015ec, 0x000015db, 0x000015ca, 0x000015b9, 0x000015a9, + 0x00001598, 0x00001588, 0x00001578, 0x00001568, 0x00001558, + 0x00001548, 0x00001538, 0x00001529, 0x0000151a, 0x0000150b, + 0x000014fc, 0x000014ed, 0x000014de, 0x000014cf, 0x000014c1, + 0x000014b3, 0x000014a5, 0x00001497, 0x00001489, 0x0000147b, + 0x0000146d, 0x00001460, 0x00001452, 0x00001445, 0x00001438, + + 0x0000142b, 0x0000141e, 0x00001411, 0x00001405, 0x000013f8, + 0x000013ec, 0x000013df, 0x000013d3, 0x000013c7, 0x000013bb, + 0x000013af, 0x000013a3, 0x00001398, 0x0000138c, 0x00001381, + 0x00001375, 0x0000136a, 0x0000135f, 0x00001354, 0x00001349, + 0x0000133e, 0x00001333, 0x00001329, 0x0000131e, 0x00001314, + 0x0000130a, 0x000012ff, 0x000012f5, 0x000012eb, 0x000012e1, + 0x000012d7, 0x000012ce, 0x000012c4, 0x000012ba, 0x000012b1, + 0x000012a7, 0x0000129e, 0x00001295, 0x0000128c, 0x00001283, + + 0x0000127a, 0x00001271, 0x00001268, 0x0000125f, 0x00001257, + 0x0000124e, 0x00001246, 0x0000123d, 0x00001235, 0x0000122d, + 0x00001224, 0x0000121c, 0x00001214, 0x0000120c, 0x00001205, + 0x000011fd, 0x000011f5, 0x000011ed, 0x000011e6, 0x000011de, + 0x000011d7, 0x000011d0, 0x000011c8, 0x000011c1, 0x000011ba, + 0x000011b3, 0x000011ac, 0x000011a5, 0x0000119e, 0x00001198, + 0x00001191, 0x0000118a, 0x00001184, 0x0000117d, 0x00001177, + 0x00001170, 0x0000116a, 0x00001164, 0x0000115e, 0x00001158, + + 0x00001151, 0x0000114c, 0x00001146, 0x00001140, 0x0000113a, + 0x00001134, 0x0000112f, 0x00001129, 0x00001123, 0x0000111e, + 0x00001119, 0x00001113, 0x0000110e, 0x00001109, 0x00001103, + 0x000010fe, 0x000010f9, 0x000010f4, 0x000010ef, 0x000010ea, + 0x000010e6, 0x000010e1, 0x000010dc, 0x000010d7, 0x000010d3, + 0x000010ce, 0x000010ca, 0x000010c5, 0x000010c1, 0x000010bd, + 0x000010b8, 0x000010b4, 0x000010b0, 0x000010ac, 0x000010a8, + 0x000010a4, 0x000010a0, 0x0000109c, 0x00001098, 0x00001094, + + 0x00001090, 0x0000108d, 0x00001089, 0x00001086, 0x00001082, + 0x0000107f, 0x0000107b, 0x00001078, 0x00001074, 0x00001071, + 0x0000106e, 0x0000106b, 0x00001068, 0x00001064, 0x00001061, + 0x0000105e, 0x0000105c, 0x00001059, 0x00001056, 0x00001053, + 0x00001050, 0x0000104e, 0x0000104b, 0x00001048, 0x00001046, + 0x00001043, 0x00001041, 0x0000103e, 0x0000103c, 0x0000103a, + 0x00001037, 0x00001035, 0x00001033, 0x00001031, 0x0000102f, + 0x0000102d, 0x0000102b, 0x00001029, 0x00001027, 0x00001025, + + 0x00001023, 0x00001022, 0x00001020, 0x0000101e, 0x0000101d, + 0x0000101b, 0x00001019, 0x00001018, 0x00001017, 0x00001015, + 0x00001014, 0x00001013, 0x00001011, 0x00001010, 0x0000100f, + 0x0000100e, 0x0000100d, 0x0000100c, 0x0000100b, 0x0000100a, + 0x00001009, 0x00001008, 0x00001007, 0x00001006, 0x00001006, + 0x00001005, 0x00001004, 0x00001004, 0x00001003, 0x00001003, + 0x00001002, 0x00001002, 0x00001001, 0x00001001, 0x00001001, + 0x00001001, 0x00001000, 0x00001000, 0x00001000, 0x00001000, + + 0x00001000 +}; diff --git a/graphics/traveler/src/trv_world.c b/graphics/traveler/src/trv_world.c new file mode 100644 index 000000000..b21379be3 --- /dev/null +++ b/graphics/traveler/src/trv_world.c @@ -0,0 +1,469 @@ +/******************************************************************************* + * apps/graphics/traveler/src/trv_world.c + * This file contains the logic that creates and destroys the world. + * + * Copyright (C) 2014 Gregory Nutt. All rights reserved. + * Author: Gregory Nutt + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in + * the documentation and/or other materials provided with the + * distribution. + * 3. Neither the name NuttX nor the names of its contributors may be + * used to endorse or promote products derived from this software + * without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, + * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, + * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS + * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED + * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN + * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + * + ****************************************************************************/ + +/**************************************************************************** + * Included files + ****************************************************************************/ + +#include "trv_types.h" +#include "trv_paltable.h" +#include "trv_world.h" +#include "trv_plane.h" +#include "trv_bitmaps.h" + +#include +#include +#include + +/**************************************************************************** + * Pre-processor Definitions + ****************************************************************************/ + +/* Everything related to the camera POV is defined in the camera section. */ + +#define CAMERA_SECTION_NAME "camera" + +/* These values define the initial camera position. */ + +#define CAMERA_INITIAL_X "initialcamerax" +#define CAMERA_INITIAL_Y "initialcameray" +#define CAMERA_INITIAL_Z "initialcameraz" + +/* These values define the orientation of the g_camera. */ + +#define CAMERA_INITIAL_YAW "initialcamerayaw" +#define CAMERA_INITIAL_PITCH "initialcamerayaw" + +/* Everything related to the player is defined in the player section. */ + +#define PLAYER_SECTION_NAME "player" + +/* These are characteristics of the player. */ + +#define PLAYER_HEIGHT "playerheight" +#define PLAYER_WALK_STEPHEIGHT "playerwalkstepheight" +#define PLAYER_RUN_STEPHEIGHT "playerrunstepheight" + +/* Everything related to the world is defined in the world section. */ + +#define WORLD_SECTION_NAME "world" + +/* Other files: */ + +#define WORLD_MAP "worldmap" +#define WORLD_PALETTE "worldpalette" +#define WORLD_IMAGES "worldimages" + +/**************************************************************************** + * Private Type Declarations + ***************************************************************************/ + +/**************************************************************************** + * Private Function Prototypes + ****************************************************************************/ + +static int trv_ini_short(INIHANDLE inihandle, FAR int16_t *value, + FAR const char *section, FAR const char *name); +#if 0 /* Not used */ +static int trv_ini_long(INIHANDLE inihandle, FAR long *value, + FAR const char *section, FAR const char *name); +#endif +static int trv_ini_filename(INIHANDLE inihandle, FAR char **filename, + FAR const char *section, FAR const char *name); +static int trv_manage_wldfile(INIHANDLE inihandle); + +/**************************************************************************** + * Public Data + ****************************************************************************/ + +/* This is the starting position and orientation of the camera in the world */ + +struct trv_camera_s g_initial_camera; + +/* This is the height of player (distance from the camera Z position to + * the position of the player's "feet" + */ + +trv_coord_t g_player_height; + +/* This is size of something that the player can step over when "walking" */ + +trv_coord_t g_walk_stepheight; + +/* This is size of something that the player can step over when "running" */ + +trv_coord_t g_run_stepheight; + +/**************************************************************************** + * Private Data + ****************************************************************************/ + +static const char g_camera_section_name[] = CAMERA_SECTION_NAME; +static const char g_camera_initialx_name[] = CAMERA_INITIAL_X; +static const char g_camera_initialy_name[] = CAMERA_INITIAL_Y; +static const char g_camera_initialz_name[] = CAMERA_INITIAL_Z; +static const char g_camera_initialyaw_name[] = CAMERA_INITIAL_YAW; +static const char g_camera_initialpitch_name[] = CAMERA_INITIAL_PITCH; + +static const char g_player_section_name[] = PLAYER_SECTION_NAME; +static const char g_player_height_name[] = PLAYER_HEIGHT; +static const char g_player_walkstepheight_name[] = PLAYER_WALK_STEPHEIGHT; +static const char g_player_runstepheight_name[] = PLAYER_RUN_STEPHEIGHT; + +static const char g_world_section_name[] = WORLD_SECTION_NAME; +static const char g_world_map_name[] = WORLD_MAP; +static const char g_world_palette_name[] = WORLD_PALETTE; +static const char g_world_images_name[] = WORLD_IMAGES; + +/**************************************************************************** + * Private Variables + ****************************************************************************/ + +/**************************************************************************** + * Private Functions + ****************************************************************************/ + +/**************************************************************************** + * Name: trv_ini_short + * + * Description: + * Reads a short value from the INI file and assures that it is within + * range for an a int16_t + * + ***************************************************************************/ + +static int trv_ini_short(INIHANDLE inihandle, FAR int16_t *value, + FAR const char *section, FAR const char *name) +{ + /* Read the long integer from the .INI file. We supply the default + * value of INT32_MAX. If this value is returned, we assume that + * that is evidence that the requested value was not supplied in the + * .INI file. + */ + + long inivalue = inifile_read_integer(inihandle, section, name, INT32_MAX); + + /* Make sure that it is in range for a int16_t. */ + + if (inivalue < INT16_MIN || inivalue > INT16_MAX) + { + /* It is not!... */ + + *value = 0; + + /* Is this because the integer was not found? or because + * it is really out of range. + */ + + if (inivalue != INT32_MAX) + { + + fprintf(stderr, "ERROR: Integer out of range in INI file:\n"); + fprintf(stderr, " Section=\"%s\" Variable name=\"%s\" value=%ld\n", + section, name, (long)inivalue); + return -ENOENT; + + } + else + { + fprintf(stderr, "ERROR: Required integer not found in INI file:\n"); + fprintf(stderr, " Section=\"%s\" Variable name=\"%s\"\n", + section, name); + return -ERANGE; + } + } + else + { + *value = (int16_t)inivalue; + return OK; + } +} + +/**************************************************************************** + * Name: trv_ini_long + * + * Description: + * Reads a long value from the INI file + * + ***************************************************************************/ + +#if 0 /* Not used */ +static uint8_t trv_ini_long(INIHANDLE inihandle, FAR long *value, + FAR const char *section, FAR const char *name) +{ + /* Read the long integer from the .INI file.*/ + + *value = inifile_read_integer(inihandle, section, name, 0); + return 0; +} +#endif + +/**************************************************************************** + * Name: trv_ini_filename + * + * Description: + * Reads a file name from the the INI file. + * + ***************************************************************************/ + +static int trv_ini_filename(INIHANDLE inihandle, FAR char **filename, + FAR const char *section, FAR const char *name) +{ + /* Read the string from the .INI file. We supply the default value of + * NULL. If this value is returned, we assume that that is evidence that + * the requested value was not supplied in the .INI file. + */ + + FAR char *value = inifile_read_string(inihandle, section, name, NULL); + + /* Did we get the file name? */ + + if (!value) + { + /* No... */ + + *filename = NULL; + + fprintf(stderr, "ERROR: Required filename not found in INI file:\n"); + fprintf(stderr, " Section=\"%s\" Variable name=\"%s\"\n", + section, name); + return -ENOENT; + } + else + { + *filename = value; + return OK; + } +} +/**************************************************************************** + * Name: trv_manage_wldfile + * + * Description: + * This is the guts of trv_world_create. It is implemented as a separate + * function in order to simplify error handling + * + ***************************************************************************/ + +static int trv_manage_wldfile(INIHANDLE inihandle) +{ + FAR char *filename; + int ret; + + /* Read the initial camera/player position */ + + ret = trv_ini_short(inihandle, &g_initial_camera.x, + g_camera_section_name, g_camera_initialx_name); + if (ret < 0) + { + return ret; + } + + ret = trv_ini_short(inihandle, &g_initial_camera.y, + g_camera_section_name, g_camera_initialy_name); + if (ret < 0) + { + return ret; + } + + ret = trv_ini_short(inihandle, &g_initial_camera.z, + g_camera_section_name, g_camera_initialz_name); + if (ret < 0) + { + return ret; + } + + /* Get the player's yaw/pitch orientation */ + + ret = trv_ini_short(inihandle, &g_initial_camera.yaw, + g_camera_section_name, g_camera_initialyaw_name); + if (ret < 0) + { + return ret; + } + + ret = trv_ini_short(inihandle, &g_initial_camera.pitch, + g_camera_section_name, g_camera_initialpitch_name); + if (ret < 0) + { + return ret; + } + + /* Get the height of the player */ + + ret = trv_ini_short(inihandle, &g_player_height, + g_player_section_name, g_player_height_name); + if (ret < 0) + { + return ret; + } + + /* Read the player's capability to step on top of things in the world. */ + + ret = trv_ini_short(inihandle, &g_walk_stepheight, + g_player_section_name, g_player_walkstepheight_name); + if (ret < 0) + { + return ret; + } + + ret = trv_ini_short(inihandle, &g_run_stepheight, + g_player_section_name, g_player_runstepheight_name); + if (ret < 0) + { + return ret; + } + + /* Get the name of the file containing the world map */ + + ret = trv_ini_filename(inihandle, &filename, + g_world_section_name, g_world_map_name); + if (ret < 0) + { + return ret; + } + + /* Allocate and load the world */ + + ret = trv_initialize_planes(); + if (ret < 0) + { + return ret; + } + + ret = trv_load_planefile(filename); + if (ret < 0) + { + return ret; + } + + inifile_free_string(filename); + + /* Get the name of the file containing the palette table which is used + * to adjust the lighting with distance. + */ + + ret = trv_ini_filename(inihandle, &filename, + g_world_section_name, g_world_palette_name); + if (ret < 0) + { + return ret; + } + + /* Then load it into g_paltable. */ + + ret = trv_load_paltable(filename); + if (ret < 0) + { + return ret; + } + + inifile_free_string(filename); + + /* Get the name of the file containing the texture data */ + + ret = trv_ini_filename(inihandle, &filename, + g_world_section_name, g_world_images_name); + if (ret < 0) + { + return ret; + } + + /* Then load the bitmaps */ + + ret = trv_initialize_bitmaps(); + if (ret < 0) + { + return ret; + } + + ret = trv_load_bitmapfile(filename); + inifile_free_string(filename); + + return ret; +} + +/**************************************************************************** + * Public Functions + ****************************************************************************/ + +/**************************************************************************** + * Name: trv_world_create + * + * Description: + * Load the world data structures from information in an INI file + * + ***************************************************************************/ + +int trv_world_create(FAR const char *wldfile) +{ + INIHANDLE inihandle; + int ret; + + /* Open the INI file which contains all of the information that we + * need to construct the world + */ + + inihandle = inifile_initialize(wldfile); + if (!inihandle) + { + fprintf(stderr, "ERROR: Could not open INI file=\"%s\"\n", wldfile); + return -ENOENT; + } + + /* Load the world file data */ + + ret = trv_manage_wldfile(inihandle); + + /* Close the INI file and return */ + + inifile_uninitialize(inihandle); + return ret; +} + +/**************************************************************************** + * Name: trv_world_destroy + * + * Description: + * Destroy the world and release all of its resources + * + ***************************************************************************/ + +void trv_world_destroy(void) +{ + trv_release_planes(); + trv_release_bitmaps(); + trv_release_paltable(); +} diff --git a/graphics/traveler/trv_romfs.c b/graphics/traveler/trv_romfs.c new file mode 100644 index 000000000..651affc03 --- /dev/null +++ b/graphics/traveler/trv_romfs.c @@ -0,0 +1,106 @@ +/**************************************************************************** + * apps/graphics/traveler/trv_romfs.c + * Mount the ROMFS demo world + * + * Copyright (C) 2014 Gregory Nutt. All rights reserved. + * Author: Gregory Nutt + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in + * the documentation and/or other materials provided with the + * distribution. + * 3. Neither the name NuttX nor the names of its contributors may be + * used to endorse or promote products derived from this software + * without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS + * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT + * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS + * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE + * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, + * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, + * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS + * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED + * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT + * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN + * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + * + ****************************************************************************/ + +/**************************************************************************** + * Included Files + ****************************************************************************/ + +#include + +#ifdef CONFIG_GRAPHICS_TRAVELER_ROMFSDEMO + +#include +#include +#include + +#include + +#include "trv_romfs.h" + +/**************************************************************************** + * Pre-processor Definitions + ****************************************************************************/ + +#ifndef CONFIG_FS_ROMFS +# error You must select CONFIG_FS_ROMFS in your configuration file +#endif + +#ifdef CONFIG_DISABLE_MOUNTPOINT +# error You must not disable mountpoints via CONFIG_DISABLE_MOUNTPOINT +#endif + +/* Describe the ROMFS file system */ + +#define SECTORSIZE 512 +#define NSECTORS(b) (((b)+SECTORSIZE-1)/SECTORSIZE) + +/**************************************************************************** + * Public Functions + ****************************************************************************/ + +/**************************************************************************** + * Name: trv_mount_world + * + * Description: + * Mount the demo world as a ROMFS file system mounted at 'mountpoint' + * + ****************************************************************************/ + +int trv_mount_world(int minor, FAR const char *mountpoint) +{ + char devpath[16]; + int ret; + + /* Create a ROM disk for the ROMFS filesystem */ + + ret = romdisk_register(minor, (FAR uint8_t *)trv_romfs_img, + NSECTORS(trv_romfs_img_len), SECTORSIZE); + if (ret < 0) + { + return ret; + } + + /* Get the ROM disk device name */ + + snprintf(devpath, 16, "/dev/ram%d", minor); + + /* Mount the file system */ + + ret = mount(devpath, mountpoint, "romfs", MS_RDONLY, NULL); + return ret; +} + +#endif /* CONFIG_GRAPHICS_TRAVELER_ROMFSDEMO */ diff --git a/graphics/traveler/world/ceil1.pcx b/graphics/traveler/world/ceil1.pcx new file mode 100644 index 000000000..115b06666 Binary files /dev/null and b/graphics/traveler/world/ceil1.pcx differ diff --git a/graphics/traveler/world/door1.pcx b/graphics/traveler/world/door1.pcx new file mode 100644 index 000000000..c477cedb2 Binary files /dev/null and b/graphics/traveler/world/door1.pcx differ diff --git a/graphics/traveler/world/door2.pcx b/graphics/traveler/world/door2.pcx new file mode 100644 index 000000000..f35d12d10 Binary files /dev/null and b/graphics/traveler/world/door2.pcx differ diff --git a/graphics/traveler/world/door5.pcx b/graphics/traveler/world/door5.pcx new file mode 100644 index 000000000..daf9457b0 Binary files /dev/null and b/graphics/traveler/world/door5.pcx differ diff --git a/graphics/traveler/world/door6.pcx b/graphics/traveler/world/door6.pcx new file mode 100644 index 000000000..61b3500de Binary files /dev/null and b/graphics/traveler/world/door6.pcx differ diff --git a/graphics/traveler/world/drkwood1.pcx b/graphics/traveler/world/drkwood1.pcx new file mode 100644 index 000000000..5e8357e8c Binary files /dev/null and b/graphics/traveler/world/drkwood1.pcx differ diff --git a/graphics/traveler/world/drkwood2.pcx b/graphics/traveler/world/drkwood2.pcx new file mode 100644 index 000000000..65f00b349 Binary files /dev/null and b/graphics/traveler/world/drkwood2.pcx differ diff --git a/graphics/traveler/world/floor1.pcx b/graphics/traveler/world/floor1.pcx new file mode 100644 index 000000000..cd8543199 Binary files /dev/null and b/graphics/traveler/world/floor1.pcx differ diff --git a/graphics/traveler/world/floor3.pcx b/graphics/traveler/world/floor3.pcx new file mode 100644 index 000000000..880add472 Binary files /dev/null and b/graphics/traveler/world/floor3.pcx differ diff --git a/graphics/traveler/world/gcloth1.pcx b/graphics/traveler/world/gcloth1.pcx new file mode 100644 index 000000000..f20fc6b86 Binary files /dev/null and b/graphics/traveler/world/gcloth1.pcx differ diff --git a/graphics/traveler/world/gcloth2.pcx b/graphics/traveler/world/gcloth2.pcx new file mode 100644 index 000000000..5529eb27e Binary files /dev/null and b/graphics/traveler/world/gcloth2.pcx differ diff --git a/graphics/traveler/world/gywall3.pcx b/graphics/traveler/world/gywall3.pcx new file mode 100644 index 000000000..86758700e Binary files /dev/null and b/graphics/traveler/world/gywall3.pcx differ diff --git a/graphics/traveler/world/gywall4.pcx b/graphics/traveler/world/gywall4.pcx new file mode 100644 index 000000000..430c612c6 Binary files /dev/null and b/graphics/traveler/world/gywall4.pcx differ diff --git a/graphics/traveler/world/marble1.pcx b/graphics/traveler/world/marble1.pcx new file mode 100644 index 000000000..0c00e784d Binary files /dev/null and b/graphics/traveler/world/marble1.pcx differ diff --git a/graphics/traveler/world/marble2.pcx b/graphics/traveler/world/marble2.pcx new file mode 100644 index 000000000..b61ef9a55 Binary files /dev/null and b/graphics/traveler/world/marble2.pcx differ diff --git a/graphics/traveler/world/mbcolm1.pcx b/graphics/traveler/world/mbcolm1.pcx new file mode 100644 index 000000000..d3711d353 Binary files /dev/null and b/graphics/traveler/world/mbcolm1.pcx differ diff --git a/graphics/traveler/world/mbcolm2.pcx b/graphics/traveler/world/mbcolm2.pcx new file mode 100644 index 000000000..6d3e2eb28 Binary files /dev/null and b/graphics/traveler/world/mbcolm2.pcx differ diff --git a/graphics/traveler/world/mbrail1.pcx b/graphics/traveler/world/mbrail1.pcx new file mode 100644 index 000000000..c0d3180d4 Binary files /dev/null and b/graphics/traveler/world/mbrail1.pcx differ diff --git a/graphics/traveler/world/mbrail2.pcx b/graphics/traveler/world/mbrail2.pcx new file mode 100644 index 000000000..9c613a910 Binary files /dev/null and b/graphics/traveler/world/mbrail2.pcx differ diff --git a/graphics/traveler/world/pcurtn1.pcx b/graphics/traveler/world/pcurtn1.pcx new file mode 100644 index 000000000..46cf844f7 Binary files /dev/null and b/graphics/traveler/world/pcurtn1.pcx differ diff --git a/graphics/traveler/world/pcurtn2.pcx b/graphics/traveler/world/pcurtn2.pcx new file mode 100644 index 000000000..d720a43d8 Binary files /dev/null and b/graphics/traveler/world/pcurtn2.pcx differ diff --git a/graphics/traveler/world/transfrm.bml b/graphics/traveler/world/transfrm.bml new file mode 100644 index 000000000..e930cdcbd --- /dev/null +++ b/graphics/traveler/world/transfrm.bml @@ -0,0 +1,16 @@ +14 +128 151 +wall1.pcx wall2.pcx +wall3.pcx wall4.pcx +gywall3.pcx gywall4.pcx +wall7.pcx wall8.pcx +floor3.pcx ceil1.pcx +drkwood1.pcx drkwood2.pcx +trose1.pcx trose2.pcx +door5.pcx door6.pcx +pcurtn1.pcx pcurtn2.pcx +wall9.pcx wall10.pcx +twndo1.pcx twndo2.pcx +marble1.pcx marble2.pcx +mbrail1.pcx mbrail2.pcx +mbcolm1.pcx mbcolm2.pcx diff --git a/graphics/traveler/world/transfrm.par b/graphics/traveler/world/transfrm.par new file mode 100644 index 000000000..953ab765b --- /dev/null +++ b/graphics/traveler/world/transfrm.par @@ -0,0 +1,23 @@ + 21 + 30 -30 1 + 32 16 1 + 48 16 1 + 64 16 1 + 80 16 1 + 96 8 1 +104 8 1 +112 8 1 +120 8 1 +128 8 1 +136 8 1 +144 8 1 +152 8 1 +160 8 1 +168 8 1 +176 8 1 +184 8 1 +192 16 1 +208 16 1 +224 16 1 +240 15 31 + diff --git a/graphics/traveler/world/transfrm.pll b/graphics/traveler/world/transfrm.pll new file mode 100644 index 000000000..8cb2a1bf1 Binary files /dev/null and b/graphics/traveler/world/transfrm.pll differ diff --git a/graphics/traveler/world/transfrm.wld b/graphics/traveler/world/transfrm.wld new file mode 100644 index 000000000..2f4e457f6 --- /dev/null +++ b/graphics/traveler/world/transfrm.wld @@ -0,0 +1,16 @@ +[camera] +InitialCameraX=1056 +InitialCameraY=1328 +InitialCameraZ=928 +InitialCameraYaw=0 +InitialCamerPitch=1893 + +[player] +PlayerHeight=352 +PlayerWalkStepHeight=126 +PlayerRunStepHeight=224 + +[world] +WorldMap=transfrm.pll +WorldPalette=transfrm.par +WorldImages=transfrm.bml diff --git a/graphics/traveler/world/trose1.pcx b/graphics/traveler/world/trose1.pcx new file mode 100644 index 000000000..625e3fb43 Binary files /dev/null and b/graphics/traveler/world/trose1.pcx differ diff --git a/graphics/traveler/world/trose2.pcx b/graphics/traveler/world/trose2.pcx new file mode 100644 index 000000000..1fcbe561d Binary files /dev/null and b/graphics/traveler/world/trose2.pcx differ diff --git a/graphics/traveler/world/twall13.pcx b/graphics/traveler/world/twall13.pcx new file mode 100644 index 000000000..e807fd9e4 Binary files /dev/null and b/graphics/traveler/world/twall13.pcx differ diff --git a/graphics/traveler/world/twndo1.pcx b/graphics/traveler/world/twndo1.pcx new file mode 100644 index 000000000..dd93eb3c4 Binary files /dev/null and b/graphics/traveler/world/twndo1.pcx differ diff --git a/graphics/traveler/world/twndo2.pcx b/graphics/traveler/world/twndo2.pcx new file mode 100644 index 000000000..a6e18e7b4 Binary files /dev/null and b/graphics/traveler/world/twndo2.pcx differ diff --git a/graphics/traveler/world/wall1.pcx b/graphics/traveler/world/wall1.pcx new file mode 100644 index 000000000..ef86ca69c Binary files /dev/null and b/graphics/traveler/world/wall1.pcx differ diff --git a/graphics/traveler/world/wall10.pcx b/graphics/traveler/world/wall10.pcx new file mode 100644 index 000000000..f15927e79 Binary files /dev/null and b/graphics/traveler/world/wall10.pcx differ diff --git a/graphics/traveler/world/wall2.pcx b/graphics/traveler/world/wall2.pcx new file mode 100644 index 000000000..f5d12be41 Binary files /dev/null and b/graphics/traveler/world/wall2.pcx differ diff --git a/graphics/traveler/world/wall3.pcx b/graphics/traveler/world/wall3.pcx new file mode 100644 index 000000000..e6d1981f0 Binary files /dev/null and b/graphics/traveler/world/wall3.pcx differ diff --git a/graphics/traveler/world/wall4.pcx b/graphics/traveler/world/wall4.pcx new file mode 100644 index 000000000..c479b075d Binary files /dev/null and b/graphics/traveler/world/wall4.pcx differ diff --git a/graphics/traveler/world/wall5.pcx b/graphics/traveler/world/wall5.pcx new file mode 100644 index 000000000..02e5cb63c Binary files /dev/null and b/graphics/traveler/world/wall5.pcx differ diff --git a/graphics/traveler/world/wall6.pcx b/graphics/traveler/world/wall6.pcx new file mode 100644 index 000000000..497148166 Binary files /dev/null and b/graphics/traveler/world/wall6.pcx differ diff --git a/graphics/traveler/world/wall7.pcx b/graphics/traveler/world/wall7.pcx new file mode 100644 index 000000000..e729b5a49 Binary files /dev/null and b/graphics/traveler/world/wall7.pcx differ diff --git a/graphics/traveler/world/wall8.pcx b/graphics/traveler/world/wall8.pcx new file mode 100644 index 000000000..a9c1ca54a Binary files /dev/null and b/graphics/traveler/world/wall8.pcx differ diff --git a/graphics/traveler/world/wall9.pcx b/graphics/traveler/world/wall9.pcx new file mode 100644 index 000000000..de38bb4dc Binary files /dev/null and b/graphics/traveler/world/wall9.pcx differ diff --git a/graphics/traveler/world/wpanel1.pcx b/graphics/traveler/world/wpanel1.pcx new file mode 100644 index 000000000..622251ca5 Binary files /dev/null and b/graphics/traveler/world/wpanel1.pcx differ diff --git a/graphics/traveler/world/wpanel2.pcx b/graphics/traveler/world/wpanel2.pcx new file mode 100644 index 000000000..392b0bf3e Binary files /dev/null and b/graphics/traveler/world/wpanel2.pcx differ