mirror of
https://github.com/wiiu-env/ftpiiu_plugin.git
synced 2025-01-08 18:10:41 +01:00
initial commit
This commit is contained in:
commit
6d4b259324
4
.gitignore
vendored
Normal file
4
.gitignore
vendored
Normal file
@ -0,0 +1,4 @@
|
||||
build/
|
||||
*.3dsx
|
||||
*.smdh
|
||||
*.elf
|
168
Makefile
Normal file
168
Makefile
Normal file
@ -0,0 +1,168 @@
|
||||
#---------------------------------------------------------------------------------
|
||||
.SUFFIXES:
|
||||
#---------------------------------------------------------------------------------
|
||||
|
||||
CTRULIB:=/home/mtheall/workspace/ninjhax/ctrulib/libctru
|
||||
|
||||
ifeq ($(strip $(DEVKITARM)),)
|
||||
$(error "Please set DEVKITARM in your environment. export DEVKITARM=<path to>devkitARM")
|
||||
endif
|
||||
|
||||
ifeq ($(strip $(CTRULIB)),)
|
||||
# THIS IS TEMPORARY - in the future it should be at $(DEVKITPRO)/libctru
|
||||
$(error "Please set CTRULIB in your environment. export CTRULIB=<path to>libctru")
|
||||
endif
|
||||
|
||||
TOPDIR ?= $(CURDIR)
|
||||
include $(DEVKITARM)/3ds_rules
|
||||
|
||||
#---------------------------------------------------------------------------------
|
||||
# TARGET is the name of the output
|
||||
# BUILD is the directory where object files & intermediate files will be placed
|
||||
# SOURCES is a list of directories containing source code
|
||||
# DATA is a list of directories containing data files
|
||||
# INCLUDES is a list of directories containing header files
|
||||
#
|
||||
# NO_SMDH: if set to anything, no SMDH file is generated.
|
||||
# APP_TITLE is the name of the app stored in the SMDH file (Optional)
|
||||
# APP_DESCRIPTION is the description of the app stored in the SMDH file (Optional)
|
||||
# APP_AUTHOR is the author of the app stored in the SMDH file (Optional)
|
||||
# ICON is the filename of the icon (.png), relative to the project folder.
|
||||
# If not set, it attempts to use one of the following (in this order):
|
||||
# - <Project name>.png
|
||||
# - icon.png
|
||||
# - <libctru folder>/default_icon.png
|
||||
#---------------------------------------------------------------------------------
|
||||
TARGET := $(notdir $(CURDIR))
|
||||
BUILD := build
|
||||
SOURCES := source
|
||||
DATA := data
|
||||
INCLUDES := include
|
||||
|
||||
APP_TITLE := ftBRONY
|
||||
APP_DESCRIPTION := Like ftPONY but magical.
|
||||
APP_AUTHOR := mtheall
|
||||
|
||||
#---------------------------------------------------------------------------------
|
||||
# options for code generation
|
||||
#---------------------------------------------------------------------------------
|
||||
ARCH := -march=armv6k -mtune=mpcore -mfloat-abi=softfp
|
||||
|
||||
CFLAGS := -g -Wall -O3 -mword-relocations \
|
||||
-fomit-frame-pointer -ffast-math \
|
||||
$(ARCH) \
|
||||
-DSTATUS_STRING="\"ftpd v1.0\""
|
||||
|
||||
CFLAGS += $(INCLUDE) -DARM11 -D_3DS
|
||||
|
||||
CXXFLAGS := $(CFLAGS) -fno-rtti -fno-exceptions -std=gnu++11
|
||||
|
||||
ASFLAGS := -g $(ARCH)
|
||||
LDFLAGS = -specs=3dsx.specs -g $(ARCH) -Wl,-Map,$(TARGET).map
|
||||
|
||||
LIBS := -lctru
|
||||
|
||||
#---------------------------------------------------------------------------------
|
||||
# list of directories containing libraries, this must be the top level containing
|
||||
# include and lib
|
||||
#---------------------------------------------------------------------------------
|
||||
LIBDIRS := $(CTRULIB)
|
||||
|
||||
|
||||
#---------------------------------------------------------------------------------
|
||||
# no real need to edit anything past this point unless you need to add additional
|
||||
# rules for different file extensions
|
||||
#---------------------------------------------------------------------------------
|
||||
ifneq ($(BUILD),$(notdir $(CURDIR)))
|
||||
#---------------------------------------------------------------------------------
|
||||
|
||||
export OUTPUT := $(CURDIR)/$(TARGET)
|
||||
export TOPDIR := $(CURDIR)
|
||||
|
||||
export VPATH := $(foreach dir,$(SOURCES),$(CURDIR)/$(dir)) \
|
||||
$(foreach dir,$(DATA),$(CURDIR)/$(dir))
|
||||
|
||||
export DEPSDIR := $(CURDIR)/$(BUILD)
|
||||
|
||||
CFILES := $(foreach dir,$(SOURCES),$(notdir $(wildcard $(dir)/*.c)))
|
||||
CPPFILES := $(foreach dir,$(SOURCES),$(notdir $(wildcard $(dir)/*.cpp)))
|
||||
SFILES := $(foreach dir,$(SOURCES),$(notdir $(wildcard $(dir)/*.s)))
|
||||
BINFILES := $(foreach dir,$(DATA),$(notdir $(wildcard $(dir)/*.*)))
|
||||
|
||||
#---------------------------------------------------------------------------------
|
||||
# use CXX for linking C++ projects, CC for standard C
|
||||
#---------------------------------------------------------------------------------
|
||||
ifeq ($(strip $(CPPFILES)),)
|
||||
export LD := $(CC)
|
||||
else
|
||||
export LD := $(CXX)
|
||||
endif
|
||||
#---------------------------------------------------------------------------------
|
||||
|
||||
export OFILES := $(addsuffix .o,$(BINFILES)) \
|
||||
$(CPPFILES:.cpp=.o) \
|
||||
$(CFILES:.c=.o) \
|
||||
$(SFILES:.s=.o)
|
||||
|
||||
export INCLUDE := $(foreach dir,$(INCLUDES),-I$(CURDIR)/$(dir)) \
|
||||
$(foreach dir,$(LIBDIRS),-I$(dir)/include) \
|
||||
-I$(CURDIR)/$(BUILD)
|
||||
|
||||
export LIBPATHS := $(foreach dir,$(LIBDIRS),-L$(dir)/lib)
|
||||
|
||||
ifeq ($(strip $(ICON)),)
|
||||
icons := $(wildcard *.png)
|
||||
ifneq (,$(findstring $(TARGET).png,$(icons)))
|
||||
export APP_ICON := $(TOPDIR)/$(TARGET).png
|
||||
else
|
||||
ifneq (,$(findstring icon.png,$(icons)))
|
||||
export APP_ICON := $(TOPDIR)/icon.png
|
||||
endif
|
||||
endif
|
||||
else
|
||||
export APP_ICON := $(TOPDIR)/$(ICON)
|
||||
endif
|
||||
|
||||
.PHONY: $(BUILD) clean all
|
||||
|
||||
#---------------------------------------------------------------------------------
|
||||
all: $(BUILD)
|
||||
|
||||
$(BUILD):
|
||||
@[ -d $@ ] || mkdir -p $@
|
||||
@make --no-print-directory -C $(BUILD) -f $(CURDIR)/Makefile
|
||||
|
||||
#---------------------------------------------------------------------------------
|
||||
clean:
|
||||
@echo clean ...
|
||||
@rm -fr $(BUILD) $(TARGET).3dsx $(OUTPUT).smdh $(TARGET).elf
|
||||
|
||||
|
||||
#---------------------------------------------------------------------------------
|
||||
else
|
||||
|
||||
DEPENDS := $(OFILES:.o=.d)
|
||||
|
||||
#---------------------------------------------------------------------------------
|
||||
# main targets
|
||||
#---------------------------------------------------------------------------------
|
||||
ifeq ($(strip $(NO_SMDH)),)
|
||||
.PHONY: all
|
||||
all : $(OUTPUT).3dsx $(OUTPUT).smdh
|
||||
endif
|
||||
$(OUTPUT).3dsx: $(OUTPUT).elf
|
||||
$(OUTPUT).elf: $(OFILES)
|
||||
|
||||
#---------------------------------------------------------------------------------
|
||||
# you need a rule like this for each extension you use as binary data
|
||||
#---------------------------------------------------------------------------------
|
||||
%.bin.o: %.bin
|
||||
#---------------------------------------------------------------------------------
|
||||
@echo $(notdir $<)
|
||||
@$(bin2o)
|
||||
|
||||
-include $(DEPENDS)
|
||||
|
||||
#---------------------------------------------------------------------------------------
|
||||
endif
|
||||
#---------------------------------------------------------------------------------------
|
BIN
data/sans.10.kerning.bin
Normal file
BIN
data/sans.10.kerning.bin
Normal file
Binary file not shown.
BIN
data/sans.10.render.bin
Normal file
BIN
data/sans.10.render.bin
Normal file
Binary file not shown.
BIN
data/sans.12.kerning.bin
Normal file
BIN
data/sans.12.kerning.bin
Normal file
Binary file not shown.
BIN
data/sans.12.render.bin
Normal file
BIN
data/sans.12.render.bin
Normal file
Binary file not shown.
BIN
data/sans.14.kerning.bin
Normal file
BIN
data/sans.14.kerning.bin
Normal file
Binary file not shown.
BIN
data/sans.14.render.bin
Normal file
BIN
data/sans.14.render.bin
Normal file
Binary file not shown.
BIN
data/sans.16.kerning.bin
Normal file
BIN
data/sans.16.kerning.bin
Normal file
Binary file not shown.
BIN
data/sans.16.render.bin
Normal file
BIN
data/sans.16.render.bin
Normal file
Binary file not shown.
BIN
data/sans.8.kerning.bin
Normal file
BIN
data/sans.8.kerning.bin
Normal file
Binary file not shown.
BIN
data/sans.8.render.bin
Normal file
BIN
data/sans.8.render.bin
Normal file
Binary file not shown.
BIN
ftbrony.png
Normal file
BIN
ftbrony.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 6.4 KiB |
12
include/console.h
Normal file
12
include/console.h
Normal file
@ -0,0 +1,12 @@
|
||||
#pragma once
|
||||
|
||||
void console_init(void);
|
||||
void console_exit(void);
|
||||
|
||||
__attribute__((format(printf,1,2)))
|
||||
void console_set_status(const char *fmt, ...);
|
||||
|
||||
__attribute__((format(printf,1,2)))
|
||||
void console_print(const char *fmt, ...);
|
||||
|
||||
void console_render(void);
|
54
include/debug.h
Normal file
54
include/debug.h
Normal file
@ -0,0 +1,54 @@
|
||||
#pragma once
|
||||
|
||||
#include <string.h>
|
||||
#ifdef _3DS
|
||||
#include <3ds.h>
|
||||
#endif
|
||||
|
||||
/*! print debug message
|
||||
*
|
||||
* @param[in] fmt format string
|
||||
* @param[in] ap varargs list
|
||||
*
|
||||
* @returns number of characters written
|
||||
*/
|
||||
static inline int
|
||||
vdebug(const char *fmt,
|
||||
va_list ap)
|
||||
{
|
||||
#ifdef _3DS
|
||||
int rc;
|
||||
char buffer[256];
|
||||
|
||||
memset(buffer, 0, sizeof(buffer));
|
||||
|
||||
/* print to buffer */
|
||||
rc = vsnprintf(buffer, sizeof(buffer)-1, fmt, ap);
|
||||
|
||||
/* call debug service with buffer */
|
||||
svcOutputDebugString(buffer, rc < sizeof(buffer) ? rc : sizeof(buffer));
|
||||
return rc;
|
||||
#else
|
||||
/* just print to stdout */
|
||||
return vprintf(fmt, ap);
|
||||
#endif
|
||||
}
|
||||
|
||||
__attribute__((format(printf,1,2)))
|
||||
/*! print debug message
|
||||
*
|
||||
* @param[in] fmt format string
|
||||
* @param[in] ... format arguments
|
||||
*
|
||||
* @returns number of characters written
|
||||
*/
|
||||
static inline int
|
||||
debug(const char *fmt, ...)
|
||||
{
|
||||
int rc;
|
||||
va_list ap;
|
||||
va_start(ap, fmt);
|
||||
rc = vdebug(fmt, ap);
|
||||
va_end(ap);
|
||||
return rc;
|
||||
}
|
5
include/ftp.h
Normal file
5
include/ftp.h
Normal file
@ -0,0 +1,5 @@
|
||||
#pragma once
|
||||
|
||||
int ftp_init(void);
|
||||
int ftp_loop(void);
|
||||
void ftp_exit(void);
|
735
source/console.c
Normal file
735
source/console.c
Normal file
@ -0,0 +1,735 @@
|
||||
#include "console.h"
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <stdarg.h>
|
||||
#ifdef _3DS
|
||||
#include <3ds.h>
|
||||
#endif
|
||||
#include "debug.h"
|
||||
|
||||
#ifdef _3DS
|
||||
#include "sans_8_kerning_bin.h"
|
||||
#include "sans_8_render_bin.h"
|
||||
#include "sans_10_kerning_bin.h"
|
||||
#include "sans_10_render_bin.h"
|
||||
#include "sans_12_kerning_bin.h"
|
||||
#include "sans_12_render_bin.h"
|
||||
#include "sans_14_kerning_bin.h"
|
||||
#include "sans_14_render_bin.h"
|
||||
#include "sans_16_kerning_bin.h"
|
||||
#include "sans_16_render_bin.h"
|
||||
|
||||
/* TODO: add support for non-ASCII characters */
|
||||
|
||||
/*! rendering information */
|
||||
typedef struct
|
||||
{
|
||||
int c; /*!< character */
|
||||
int y_off; /*!< vertical offset */
|
||||
int width; /*!< width */
|
||||
int height; /*!< height */
|
||||
int x_adv; /*!< horizontal advance */
|
||||
u8 data[]; /*!< width*height bitmap */
|
||||
} render_info_t;
|
||||
|
||||
/*! kerning information */
|
||||
typedef struct
|
||||
{
|
||||
int prev; /*!< previous character */
|
||||
int next; /*!< next character */
|
||||
int x_off; /*!< horizontal adjustment */
|
||||
} kerning_info_t;
|
||||
|
||||
/*! font data */
|
||||
typedef struct
|
||||
{
|
||||
const char *name; /*!< font name */
|
||||
render_info_t **render_info; /*!< render information list */
|
||||
kerning_info_t *kerning_info; /*!< kerning information list */
|
||||
size_t num_render_info; /*!< number of render information nodes */
|
||||
size_t num_kerning_info; /*!< number of kerning information nodes */
|
||||
int pt; /*!< font size */
|
||||
} font_t;
|
||||
|
||||
/*! font information */
|
||||
typedef struct
|
||||
{
|
||||
const char *name; /*!< font name */
|
||||
int pt; /*!< font size */
|
||||
const u8 *render_data; /*!< render data */
|
||||
const u8 *kerning_data; /*!< kerning data */
|
||||
const u32 *render_data_size; /*!< render data size */
|
||||
const u32 *kerning_data_size; /*!< kerning data size */
|
||||
} font_info_t;
|
||||
|
||||
/*! font descriptors */
|
||||
static font_info_t font_info[] =
|
||||
{
|
||||
#define FONT_INFO(name, pt) \
|
||||
{ #name, pt, name##_##pt##_render_bin, name##_##pt##_kerning_bin, \
|
||||
&name##_##pt##_render_bin_size, &name##_##pt##_kerning_bin_size, }
|
||||
FONT_INFO(sans, 8),
|
||||
FONT_INFO(sans, 10),
|
||||
FONT_INFO(sans, 12),
|
||||
FONT_INFO(sans, 14),
|
||||
FONT_INFO(sans, 16),
|
||||
};
|
||||
/*! number of font descriptors */
|
||||
static const size_t num_font_info = sizeof(font_info)/sizeof(font_info[0]);
|
||||
|
||||
/*! find next render info
|
||||
*
|
||||
* @param[in] info current render info
|
||||
*
|
||||
* @returns next render info
|
||||
*/
|
||||
static render_info_t*
|
||||
next_render_info(render_info_t *info)
|
||||
{
|
||||
char *ptr = (char*)info;
|
||||
ptr += sizeof(*info) + info->width*info->height;
|
||||
ptr = (char*)(((int)ptr + sizeof(int)-1) & ~(sizeof(int)-1));
|
||||
|
||||
return (render_info_t*)ptr;
|
||||
}
|
||||
|
||||
/*! free font info
|
||||
*
|
||||
* @param[in] font
|
||||
*/
|
||||
static void
|
||||
free_font(font_t *font)
|
||||
{
|
||||
free(font->render_info);
|
||||
free(font);
|
||||
}
|
||||
|
||||
/*! load font info
|
||||
*
|
||||
* @param[in] name
|
||||
* @param[in] pt
|
||||
* @param[in] render_data
|
||||
* @param[in] render_data_size
|
||||
* @param[in] kerning data
|
||||
* @param[in] kerning_data_size
|
||||
*
|
||||
* @returns font info
|
||||
*/
|
||||
static font_t*
|
||||
load_font(const char *name,
|
||||
int pt,
|
||||
const u8 *render_data,
|
||||
size_t render_data_size,
|
||||
const u8 *kerning_data,
|
||||
size_t kerning_data_size)
|
||||
{
|
||||
size_t i;
|
||||
render_info_t *rinfo;
|
||||
font_t *font;
|
||||
|
||||
/* allocate new font info */
|
||||
font = (font_t*)calloc(1, sizeof(font_t));
|
||||
if(font != NULL)
|
||||
{
|
||||
/* count number of render entries */
|
||||
rinfo = (render_info_t*)render_data;
|
||||
while((u8*)rinfo < render_data + render_data_size)
|
||||
{
|
||||
++font->num_render_info;
|
||||
rinfo = next_render_info(rinfo);
|
||||
}
|
||||
|
||||
/* allocate array of render info pointers */
|
||||
font->render_info = (render_info_t**)calloc(font->num_render_info, sizeof(render_info_t));
|
||||
if(font->render_info != NULL)
|
||||
{
|
||||
/* fill in the pointer list */
|
||||
rinfo = (render_info_t*)render_data;
|
||||
i = 0;
|
||||
while((u8*)rinfo < render_data + render_data_size)
|
||||
{
|
||||
font->render_info[i++] = rinfo;
|
||||
rinfo = next_render_info(rinfo);
|
||||
}
|
||||
|
||||
/* fill in the kerning info */
|
||||
font->kerning_info = (kerning_info_t*)kerning_data;
|
||||
font->num_kerning_info = kerning_data_size / sizeof(kerning_info_t);
|
||||
|
||||
/* set font size and name */
|
||||
font->pt = pt;
|
||||
font->name = name;
|
||||
}
|
||||
else
|
||||
{
|
||||
/* failed to allocate render info list */
|
||||
free_font(font);
|
||||
font = NULL;
|
||||
}
|
||||
}
|
||||
|
||||
return font;
|
||||
}
|
||||
|
||||
/*! list of font info entries */
|
||||
static font_t **fonts;
|
||||
/*! number of font info entries */
|
||||
static size_t num_fonts = 0;
|
||||
|
||||
/*! compare two fonts
|
||||
*
|
||||
* @param[in] p1 left side of comparison (font_t**)
|
||||
* @param[in] p2 right side of comparison (font_t**)
|
||||
*
|
||||
* @returns <0 if p1 < p2
|
||||
* @returns 0 if p1 == p2
|
||||
* @returns >0 if p1 > p2
|
||||
*/
|
||||
static int
|
||||
font_cmp(const void *p1,
|
||||
const void *p2)
|
||||
{
|
||||
/* interpret parameters */
|
||||
font_t *f1 = *(font_t**)p1;
|
||||
font_t *f2 = *(font_t**)p2;
|
||||
|
||||
/* major key is font name */
|
||||
int rc = strcmp(f1->name, f2->name);
|
||||
if(rc != 0)
|
||||
return rc;
|
||||
|
||||
/* minor key is font size */
|
||||
if(f1->pt < f2->pt)
|
||||
return -1;
|
||||
if(f1->pt > f2->pt)
|
||||
return 1;
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*! search for a font by name and size
|
||||
*
|
||||
* @param[in] name font name
|
||||
* @param[in] pt font size
|
||||
*
|
||||
* @returns matching font
|
||||
*/
|
||||
static font_t*
|
||||
find_font(const char *name,
|
||||
int pt)
|
||||
{
|
||||
/* create a key to search for */
|
||||
font_t key, *keyptr;
|
||||
key.name = name;
|
||||
key.pt = pt;
|
||||
keyptr = &key;
|
||||
|
||||
/* search for the key */
|
||||
void *font = bsearch(&keyptr, fonts, num_fonts, sizeof(font_t*), font_cmp);
|
||||
if(font == NULL)
|
||||
return NULL;
|
||||
|
||||
/* found it */
|
||||
return *(font_t**)font;
|
||||
}
|
||||
|
||||
/*! initialize console subsystem */
|
||||
void
|
||||
console_init(void)
|
||||
{
|
||||
size_t i;
|
||||
|
||||
/* allocate font list */
|
||||
fonts = (font_t**)calloc(num_font_info, sizeof(font_t*));
|
||||
if(fonts == NULL)
|
||||
return;
|
||||
|
||||
/* load fonts */
|
||||
for(i = 0; i < num_font_info; ++i)
|
||||
{
|
||||
font_info_t *info = &font_info[i];
|
||||
fonts[num_fonts] = load_font(info->name, info->pt,
|
||||
info->render_data,
|
||||
*info->render_data_size,
|
||||
info->kerning_data,
|
||||
*info->kerning_data_size);
|
||||
if(fonts[num_fonts] != NULL)
|
||||
++num_fonts;
|
||||
}
|
||||
|
||||
/* sort the list for bsearch later */
|
||||
qsort(fonts, num_fonts, sizeof(font_t*), font_cmp);
|
||||
}
|
||||
|
||||
/*! deinitialize console subsystem */
|
||||
void
|
||||
console_exit(void)
|
||||
{
|
||||
int i;
|
||||
|
||||
/* free the font info */
|
||||
for(i = 0; i < num_fonts; ++i)
|
||||
free_font(fonts[i]);
|
||||
|
||||
/* free the font info list */
|
||||
free(fonts);
|
||||
fonts = NULL;
|
||||
}
|
||||
|
||||
/*! status bar contents */
|
||||
static char status[64];
|
||||
/*! console buffer */
|
||||
static char buffer[8192];
|
||||
/*! pointer to end of buffer */
|
||||
static char *buffer_end = buffer + sizeof(buffer);
|
||||
/*! pointer to end of console contents */
|
||||
static char *end = buffer;
|
||||
|
||||
/*! count lines in console contents */
|
||||
static size_t
|
||||
count_lines(void)
|
||||
{
|
||||
size_t lines = 0;
|
||||
char *p = buffer;
|
||||
|
||||
/* search for each newline character */
|
||||
while(p < end && (p = strchr(p, '\n')) != NULL)
|
||||
{
|
||||
++lines;
|
||||
++p;
|
||||
}
|
||||
|
||||
return lines;
|
||||
}
|
||||
|
||||
/*! remove lines that have "scrolled" off screen */
|
||||
static void
|
||||
reduce_lines(void)
|
||||
{
|
||||
int lines = count_lines();
|
||||
char *p = buffer;
|
||||
|
||||
/* we can fit 18 lines on the screen */
|
||||
/* TODO make based on pt size */
|
||||
while(lines > 18)
|
||||
{
|
||||
p = strchr(p, '\n');
|
||||
++p;
|
||||
--lines;
|
||||
}
|
||||
|
||||
/* move the new beginning to where it needs to be */
|
||||
ptrdiff_t distance = p - buffer;
|
||||
memmove(buffer, buffer+distance, end - p);
|
||||
end -= distance;
|
||||
*end = 0;
|
||||
}
|
||||
|
||||
/*! set status bar contents
|
||||
*
|
||||
* @param[in] fmt format string
|
||||
* @param[in] ... format arguments
|
||||
*/
|
||||
void
|
||||
console_set_status(const char *fmt, ...)
|
||||
{
|
||||
va_list ap;
|
||||
|
||||
va_start(ap, fmt);
|
||||
memset(status, 0, sizeof(status));
|
||||
vsnprintf(status, sizeof(status)-1, fmt, ap);
|
||||
va_end(ap);
|
||||
}
|
||||
|
||||
/*! add text to the console
|
||||
*
|
||||
* @param[in] fmt format string
|
||||
* @param[in] ... format arguments
|
||||
*/
|
||||
void
|
||||
console_print(const char *fmt, ...)
|
||||
{
|
||||
int rc;
|
||||
va_list ap;
|
||||
|
||||
/* append to the end of the console buffer */
|
||||
va_start(ap, fmt);
|
||||
rc = vsnprintf(end, buffer_end - end - 1, fmt, ap);
|
||||
va_end(ap);
|
||||
|
||||
/* null terminate buffer */
|
||||
end += rc;
|
||||
if(end >= buffer_end)
|
||||
end = buffer_end - 1;
|
||||
*end = 0;
|
||||
|
||||
/* scroll */
|
||||
reduce_lines();
|
||||
}
|
||||
|
||||
/*! compare render information
|
||||
*
|
||||
* @param[in] p1 left side of comparison (render_info_t**)
|
||||
* @param[in] p2 right side of comparison (render_info_t**)
|
||||
*
|
||||
* @returns <0 if p1 < p2
|
||||
* @returns 0 if p1 == p2
|
||||
* @returns >0 if p1 > p2
|
||||
*/
|
||||
static int
|
||||
render_info_cmp(const void *p1,
|
||||
const void *p2)
|
||||
{
|
||||
/* interpret parameters */
|
||||
render_info_t *r1 = *(render_info_t**)p1;
|
||||
render_info_t *r2 = *(render_info_t**)p2;
|
||||
|
||||
/* ordered by character */
|
||||
if(r1->c < r2->c)
|
||||
return -1;
|
||||
else if(r1->c > r2->c)
|
||||
return 1;
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*! search for render info by character
|
||||
*
|
||||
* @param[in] font font info
|
||||
* @param[in] char character
|
||||
*
|
||||
* @returns matching render info
|
||||
*/
|
||||
static render_info_t*
|
||||
find_render_info(font_t *font,
|
||||
char c)
|
||||
{
|
||||
/* create a key to search for */
|
||||
render_info_t key, *keyptr;
|
||||
key.c = c;
|
||||
keyptr = &key;
|
||||
|
||||
/* search for the key */
|
||||
void *info = bsearch(&keyptr, font->render_info, font->num_render_info,
|
||||
sizeof(render_info_t*), render_info_cmp);
|
||||
if(info == NULL)
|
||||
return NULL;
|
||||
|
||||
/* found it */
|
||||
return *(render_info_t**)info;
|
||||
}
|
||||
|
||||
/*! compare kerning information
|
||||
*
|
||||
* @param[in] p1 left side of comparison (kerning_info_t*)
|
||||
* @param[in] p2 right side of comparison (kerning_info_t*)
|
||||
*
|
||||
* @returns <0 if p1 < p2
|
||||
* @returns 0 if p1 == p2
|
||||
* @returns >0 if p1 > p2
|
||||
*/
|
||||
static int
|
||||
kerning_info_cmp(const void *p1,
|
||||
const void *p2)
|
||||
{
|
||||
/* interpret parameters */
|
||||
kerning_info_t *k1 = (kerning_info_t*)p1;
|
||||
kerning_info_t *k2 = (kerning_info_t*)p2;
|
||||
|
||||
/* major key is prev */
|
||||
if(k1->prev < k2->prev)
|
||||
return -1;
|
||||
if(k1->prev > k2->prev)
|
||||
return 1;
|
||||
|
||||
/* minor key is next */
|
||||
if(k1->next < k2->next)
|
||||
return -1;
|
||||
if(k1->next > k2->next)
|
||||
return 1;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*! search for kerning info by character pair
|
||||
*
|
||||
* @param[in] font font info
|
||||
* @param[in] prev prev character
|
||||
* @param[in] next next character
|
||||
*
|
||||
* @returns matching render info
|
||||
*/
|
||||
static kerning_info_t*
|
||||
find_kerning_info(font_t *font,
|
||||
char prev,
|
||||
char next)
|
||||
{
|
||||
/* create a key to search for */
|
||||
kerning_info_t key;
|
||||
key.prev = prev;
|
||||
key.next = next;
|
||||
|
||||
/* search for the key */
|
||||
void *info = bsearch(&key, font->kerning_info, font->num_kerning_info,
|
||||
sizeof(kerning_info_t), kerning_info_cmp);
|
||||
if(info == NULL)
|
||||
return NULL;
|
||||
|
||||
/* found it */
|
||||
return (kerning_info_t*)info;
|
||||
}
|
||||
|
||||
/*! clear framebuffer
|
||||
*
|
||||
* @param[in] screen screen to clear
|
||||
* @param[in] side which side on the stereoscopic display
|
||||
* @param[in] rgbColor clear color
|
||||
*/
|
||||
static void
|
||||
clear_screen(gfxScreen_t screen,
|
||||
gfx3dSide_t side,
|
||||
u8 rgbColor[3])
|
||||
{
|
||||
/* get the framebuffer information */
|
||||
u16 fbWidth, fbHeight;
|
||||
u8 *fb = gfxGetFramebuffer(screen, side, &fbWidth, &fbHeight);
|
||||
|
||||
/* fill the framebuffer with the clear color */
|
||||
int i;
|
||||
for(i = 0; i < fbWidth*fbHeight; ++i)
|
||||
{
|
||||
*(fb++) = rgbColor[2];
|
||||
*(fb++) = rgbColor[1];
|
||||
*(fb++) = rgbColor[0];
|
||||
}
|
||||
}
|
||||
|
||||
/*! draw a quad
|
||||
*
|
||||
* @param[in] screen screen to draw to
|
||||
* @param[in] side which side on the stereoscopic display
|
||||
* @param[in] data quad data
|
||||
* @param[in] x quad x position
|
||||
* @param[in] y quad y position
|
||||
* @param[in] w quad width
|
||||
* @param[in] h quad height
|
||||
*
|
||||
* @note this quad data is 8-bit alpha-only
|
||||
* @note uses framebuffer native coordinates
|
||||
*/
|
||||
static void
|
||||
draw_quad(gfxScreen_t screen,
|
||||
gfx3dSide_t side,
|
||||
const u8 *data,
|
||||
int x,
|
||||
int y,
|
||||
int w,
|
||||
int h)
|
||||
{
|
||||
int i, j;
|
||||
int index = 0;
|
||||
int stride = w;
|
||||
|
||||
/* get the framebuffer information */
|
||||
u16 width, height;
|
||||
u8 *fb = gfxGetFramebuffer(screen, side, &width, &height);
|
||||
|
||||
/* this quad is totally offscreen; don't draw */
|
||||
if(x > width || y > height)
|
||||
return;
|
||||
|
||||
/* this quad is totally offscreen; don't draw */
|
||||
if(x + w < 0 || y + h < 0)
|
||||
return;
|
||||
|
||||
/* adjust parameters for partially visible quad */
|
||||
if(x < 0)
|
||||
{
|
||||
index -= x;
|
||||
w += x;
|
||||
x = 0;
|
||||
}
|
||||
|
||||
/* adjust parameters for partially visible quad */
|
||||
if(y < 0)
|
||||
{
|
||||
index -= y*stride;
|
||||
h += y;
|
||||
y = 0;
|
||||
}
|
||||
|
||||
/* adjust parameters for partially visible quad */
|
||||
if(x + w > width)
|
||||
w = width - x;
|
||||
|
||||
/* adjust parameters for partially visible quad */
|
||||
if(y + h > height)
|
||||
h = height - y;
|
||||
|
||||
/* move framebuffer pointer to quad start position */
|
||||
fb += (y*width + x)*3;
|
||||
|
||||
/* fill in data */
|
||||
for(j = 0; j < h; ++j)
|
||||
{
|
||||
for(i = 0; i < w; ++i)
|
||||
{
|
||||
/* alpha blending; assuming color is white */
|
||||
int v = data[index];
|
||||
fb[0] = fb[0]*(0xFF-v)/0xFF + v;
|
||||
fb[1] = fb[1]*(0xFF-v)/0xFF + v;
|
||||
fb[2] = fb[2]*(0xFF-v)/0xFF + v;
|
||||
|
||||
++index;
|
||||
fb += 3;
|
||||
}
|
||||
|
||||
index += (stride-w);
|
||||
fb += (width-w)*3;
|
||||
}
|
||||
}
|
||||
|
||||
/*! draw text to framebuffer
|
||||
*
|
||||
* @param[in] screen screen to draw to
|
||||
* @param[in] side which side on the stereoscopic display
|
||||
* @param[in] font font to use when rendering
|
||||
* @param[in] data quad data
|
||||
* @param[in] x quad x position
|
||||
* @param[in] y quad y position
|
||||
*
|
||||
* @note uses intuitive coordinates
|
||||
*/
|
||||
static void
|
||||
draw_text(gfxScreen_t screen,
|
||||
gfx3dSide_t side,
|
||||
font_t *font,
|
||||
const char *data,
|
||||
int x,
|
||||
int y)
|
||||
{
|
||||
render_info_t *rinfo;
|
||||
kerning_info_t *kinfo;
|
||||
const char *p;
|
||||
int xoff = x, yoff = y;
|
||||
char prev = 0;
|
||||
|
||||
/* draw each character */
|
||||
for(p = data; *p != 0; ++p)
|
||||
{
|
||||
/* newline; move down a line and all the way left */
|
||||
if(*p == '\n')
|
||||
{
|
||||
xoff = x;
|
||||
yoff += font->pt + font->pt/2;
|
||||
prev = 0;
|
||||
continue;
|
||||
}
|
||||
|
||||
/* look up the render info for this character */
|
||||
rinfo = find_render_info(font, *p);
|
||||
|
||||
/* couldn't find it; just ignore it */
|
||||
if(rinfo == NULL)
|
||||
continue;
|
||||
|
||||
/* find kerning data */
|
||||
kinfo = NULL;
|
||||
if(prev != 0)
|
||||
kinfo = find_kerning_info(font, prev, *p);
|
||||
|
||||
/* adjust for kerning */
|
||||
if(kinfo != NULL)
|
||||
xoff += kinfo->x_off >> 6;
|
||||
|
||||
/* save this character for next kerning lookup */
|
||||
prev = *p;
|
||||
|
||||
/* render character */
|
||||
if(rinfo->width != 0 && rinfo->height != 0)
|
||||
{
|
||||
int x, y;
|
||||
|
||||
/* get framebuffer info */
|
||||
u16 width, height;
|
||||
gfxGetFramebuffer(screen, side, &width, &height);
|
||||
|
||||
/* transform intuitive coordinates to framebuffer-native */
|
||||
x = width - yoff - font->pt - 2 - (rinfo->height - rinfo->y_off);
|
||||
y = xoff;
|
||||
|
||||
/* draw character */
|
||||
draw_quad(screen, side, rinfo->data, x, y,
|
||||
rinfo->height, rinfo->width);
|
||||
}
|
||||
|
||||
/* advance to next character coordinate */
|
||||
xoff += rinfo->x_adv >> 6;
|
||||
}
|
||||
}
|
||||
|
||||
/*! draw console to screen */
|
||||
void
|
||||
console_render(void)
|
||||
{
|
||||
font_t *font;
|
||||
|
||||
/* clear all screens */
|
||||
u8 bluish[] = { 0, 0, 127 };
|
||||
clear_screen(GFX_TOP, GFX_LEFT, bluish);
|
||||
clear_screen(GFX_BOTTOM, GFX_LEFT, bluish);
|
||||
|
||||
/* look up font for status bar and draw status bar */
|
||||
font = find_font("sans", 10);
|
||||
if(font != NULL)
|
||||
draw_text(GFX_TOP, GFX_LEFT, font, status, 4, 4);
|
||||
else
|
||||
debug("%s: couldn't find 'sans 10pt'\n", __func__);
|
||||
|
||||
/* look up font for console and draw console */
|
||||
font = find_font("sans", 8);
|
||||
if(font != NULL)
|
||||
draw_text(GFX_TOP, GFX_LEFT, font, buffer, 4, 20);
|
||||
else
|
||||
debug("%s: couldn't find 'sans 8pt'\n", __func__);
|
||||
|
||||
/* flush framebuffer */
|
||||
gfxFlushBuffers();
|
||||
gspWaitForVBlank();
|
||||
gfxSwapBuffers();
|
||||
}
|
||||
#else
|
||||
|
||||
/* this is a lot easier when you have a real console */
|
||||
|
||||
void
|
||||
console_init(void)
|
||||
{
|
||||
}
|
||||
|
||||
void
|
||||
console_exit(void)
|
||||
{
|
||||
}
|
||||
|
||||
void
|
||||
console_set_status(const char *fmt, ...)
|
||||
{
|
||||
va_list ap;
|
||||
va_start(ap, fmt);
|
||||
vprintf(fmt, ap);
|
||||
va_end(ap);
|
||||
fputc('\n', stdout);
|
||||
}
|
||||
|
||||
void
|
||||
console_print(const char *fmt, ...)
|
||||
{
|
||||
va_list ap;
|
||||
va_start(ap, fmt);
|
||||
vprintf(fmt, ap);
|
||||
va_end(ap);
|
||||
}
|
||||
|
||||
void console_render(void)
|
||||
{
|
||||
}
|
||||
#endif
|
2136
source/ftp.c
Normal file
2136
source/ftp.c
Normal file
File diff suppressed because it is too large
Load Diff
114
source/main.c
Normal file
114
source/main.c
Normal file
@ -0,0 +1,114 @@
|
||||
#include <malloc.h>
|
||||
#include <stdarg.h>
|
||||
#include <stdio.h>
|
||||
#ifdef _3DS
|
||||
#include <3ds.h>
|
||||
#endif
|
||||
#include "console.h"
|
||||
#include "ftp.h"
|
||||
|
||||
/*! looping mechanism
|
||||
*
|
||||
* @param[in] callback function to call during each iteration
|
||||
*/
|
||||
static void
|
||||
loop(int (*callback)(void))
|
||||
{
|
||||
#ifdef _3DS
|
||||
int rc;
|
||||
APP_STATUS status;
|
||||
|
||||
/* check apt status */
|
||||
while((status = aptGetStatus()) != APP_EXITING)
|
||||
{
|
||||
rc = 0;
|
||||
if(status == APP_RUNNING)
|
||||
rc = callback();
|
||||
else if(status == APP_SUSPENDING)
|
||||
aptReturnToMenu();
|
||||
else if(status == APP_SLEEPMODE)
|
||||
aptWaitStatusEvent();
|
||||
|
||||
if(rc == 0)
|
||||
console_render();
|
||||
else
|
||||
return;
|
||||
}
|
||||
#else
|
||||
for(;;)
|
||||
callback();
|
||||
#endif
|
||||
}
|
||||
|
||||
/*! wait until the B button is pressed
|
||||
*
|
||||
* @returns -1 if B was pressed
|
||||
*/
|
||||
static int
|
||||
wait_for_b(void)
|
||||
{
|
||||
#ifdef _3DS
|
||||
/* update button state */
|
||||
hidScanInput();
|
||||
|
||||
/* check if B was pressed */
|
||||
if(hidKeysDown() & KEY_B)
|
||||
return -1;
|
||||
|
||||
/* B was not pressed */
|
||||
return 0;
|
||||
#else
|
||||
return -1;
|
||||
#endif
|
||||
}
|
||||
|
||||
/*! entry point
|
||||
*
|
||||
* @param[in] argc unused
|
||||
* @param[in] argv unused
|
||||
*
|
||||
* returns exit status
|
||||
*/
|
||||
int
|
||||
main(int argc,
|
||||
char *argv[])
|
||||
{
|
||||
#ifdef _3DS
|
||||
/* initialize needed 3DS services */
|
||||
srvInit();
|
||||
aptInit();
|
||||
hidInit(NULL);
|
||||
irrstInit(NULL);
|
||||
gfxInit();
|
||||
gfxSet3D(false);
|
||||
#endif
|
||||
|
||||
/* initialize console subsystem */
|
||||
console_init();
|
||||
console_set_status(STATUS_STRING);
|
||||
|
||||
/* initialize ftp subsystem */
|
||||
if(ftp_init() == 0)
|
||||
{
|
||||
/* ftp loop */
|
||||
loop(ftp_loop);
|
||||
|
||||
/* done with ftp */
|
||||
ftp_exit();
|
||||
}
|
||||
|
||||
console_print("Press B to exit\n");
|
||||
loop(wait_for_b);
|
||||
console_exit();
|
||||
|
||||
#ifdef _3DS
|
||||
/* deinitialize 3DS services */
|
||||
gfxExit();
|
||||
irrstExit();
|
||||
hidExit();
|
||||
aptExit();
|
||||
srvExit();
|
||||
#endif
|
||||
|
||||
return 0;
|
||||
}
|
Loading…
Reference in New Issue
Block a user