mirror of
https://github.com/wiiu-env/ftpiiu_plugin.git
synced 2025-01-25 17:41:17 +01:00
first commit
This commit is contained in:
commit
51247b0dc1
4
.gitignore
vendored
Normal file
4
.gitignore
vendored
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
build/*
|
||||||
|
*.mod
|
||||||
|
sysapp.layout
|
||||||
|
sysapp.cbp
|
272
Makefile
Normal file
272
Makefile
Normal file
@ -0,0 +1,272 @@
|
|||||||
|
# You probably never need to adjust this Makefile.
|
||||||
|
# All changes can be done in the makefile.mk
|
||||||
|
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
# Clear the implicit built in rules
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
.SUFFIXES:
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
ifeq ($(strip $(DEVKITPPC)),)
|
||||||
|
$(error "Please set DEVKITPPC in your environment. export DEVKITPPC=<path to>devkitPPC")
|
||||||
|
endif
|
||||||
|
ifeq ($(strip $(DEVKITPRO)),)
|
||||||
|
$(error "Please set DEVKITPRO in your environment. export DEVKITPRO=<path to>devkitPRO")
|
||||||
|
endif
|
||||||
|
|
||||||
|
export PATH := $(DEVKITPPC)/bin:$(PORTLIBS)/bin:$(PATH)
|
||||||
|
export PORTLIBS := $(DEVKITPRO)/portlibs/ppc
|
||||||
|
export WUPSDIR := $(DEVKITPRO)/wups
|
||||||
|
export GCC_VER := $(shell $(DEVKITPPC)/bin/powerpc-eabi-gcc -dumpversion)
|
||||||
|
|
||||||
|
PREFIX := powerpc-eabi-
|
||||||
|
|
||||||
|
export AS := $(PREFIX)as
|
||||||
|
export CC := $(PREFIX)gcc
|
||||||
|
export CXX := $(PREFIX)g++
|
||||||
|
export LD := $(PREFIX)ld
|
||||||
|
export AR := $(PREFIX)ar
|
||||||
|
export OBJCOPY := $(PREFIX)objcopy
|
||||||
|
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
# 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
|
||||||
|
# INCLUDES is a list of directories containing extra header files
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
TARGET := $(notdir $(CURDIR))
|
||||||
|
BUILD := build
|
||||||
|
|
||||||
|
ifeq ($(notdir $(CURDIR)),$(BUILD))
|
||||||
|
include ../makefile.mk
|
||||||
|
else
|
||||||
|
include makefile.mk
|
||||||
|
endif
|
||||||
|
|
||||||
|
include $(WUPSDIR)/plugin_makefile.mk
|
||||||
|
|
||||||
|
|
||||||
|
#MAP ?= $(TARGET:.mod=.map)
|
||||||
|
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
# options for code generation
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
MACHDEP = -DESPRESSO -mcpu=750 -meabi -mhard-float
|
||||||
|
|
||||||
|
# -Os: optimise size
|
||||||
|
# -Wall: generate lots of warnings
|
||||||
|
# -D__wiiu__: define the symbol __wiiu__ (used in some headers)
|
||||||
|
# -mcpu=750: enable processor specific compilation
|
||||||
|
# -meabi: enable eabi specific compilation
|
||||||
|
# -mhard-float: enable hardware floating point instructions
|
||||||
|
# -nostartfiles: Do not use the standard system startup files when linking
|
||||||
|
# -ffunction-sections: split up functions so linker can garbage collect
|
||||||
|
# -fdata-sections: split up data so linker can garbage collect
|
||||||
|
COMMON_CFLAGS := -O0 -Wall $(MACHDEP) -meabi -ffunction-sections -fdata-sections -Wl,-q $(COMMON_CFLAGS)
|
||||||
|
|
||||||
|
CFLAGS += -D__WIIU__ -D__WUT__
|
||||||
|
|
||||||
|
# -x c: compile as c code
|
||||||
|
# -std=c11: use the c11 standard
|
||||||
|
CFLAGS := $(COMMON_CFLAGS) -x c -std=gnu11 $(CFLAGS)
|
||||||
|
|
||||||
|
# -x c++: compile as c++ code
|
||||||
|
# -std=gnu++11: use the c++11 standard
|
||||||
|
CXXFLAGS := $(COMMON_CFLAGS) -x c++ -std=gnu++11 $(CXXFLAGS)
|
||||||
|
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
# any extra ld flags
|
||||||
|
#--------------------------------------------------------------------------------
|
||||||
|
# --gc-sections: remove unneeded symbols
|
||||||
|
# -Map: generate a map file
|
||||||
|
LDFLAGS += $(ARCH) -Wl,-Map,$(notdir $@).map,--gc-sections,-wrap,__gxx_personality_v0
|
||||||
|
|
||||||
|
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
Q := @
|
||||||
|
MAKEFLAGS += --no-print-directory
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
# any extra libraries we wish to link with the project
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
LIBS +=
|
||||||
|
#
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
# list of directories containing libraries, this must be the top level containing
|
||||||
|
# include and lib
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
LIBDIRS +=
|
||||||
|
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
# 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 PROJECTDIR := $(CURDIR)
|
||||||
|
export OUTPUT := $(CURDIR)/$(TARGETDIR)/$(TARGET)
|
||||||
|
export VPATH := $(foreach dir,$(SOURCES),$(CURDIR)/$(dir)) \
|
||||||
|
$(foreach dir,$(DATA),$(CURDIR)/$(dir))
|
||||||
|
export DEPSDIR := $(CURDIR)/$(BUILD)
|
||||||
|
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
# automatically build a list of object files for our project
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
CFILES := $(foreach dir,$(SOURCES),$(notdir $(wildcard $(dir)/*.c)))
|
||||||
|
CPPFILES := $(foreach dir,$(SOURCES),$(notdir $(wildcard $(dir)/*.cpp)))
|
||||||
|
sFILES := $(foreach dir,$(SOURCES),$(notdir $(wildcard $(dir)/*.s)))
|
||||||
|
SFILES := $(foreach dir,$(SOURCES),$(notdir $(wildcard $(dir)/*.S)))
|
||||||
|
BINFILES := $(foreach dir,$(DATA),$(notdir $(wildcard $(dir)/*.*)))
|
||||||
|
TTFFILES := $(foreach dir,$(DATA),$(notdir $(wildcard $(dir)/*.ttf)))
|
||||||
|
PNGFILES := $(foreach dir,$(SOURCES),$(notdir $(wildcard $(dir)/*.png)))
|
||||||
|
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
# use CXX for linking C++ projects, CC for standard C
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
ifeq ($(strip $(CPPFILES)),)
|
||||||
|
export REAL_LD := $(CC)
|
||||||
|
else
|
||||||
|
export REAL_LD := $(CXX)
|
||||||
|
endif
|
||||||
|
|
||||||
|
export OFILES := $(CPPFILES:.cpp=.o) $(CFILES:.c=.o) \
|
||||||
|
$(sFILES:.s=.o) $(SFILES:.S=.o) \
|
||||||
|
$(PNGFILES:.png=.png.o) $(addsuffix .o,$(BINFILES))
|
||||||
|
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
# build a list of include paths
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
export INCLUDE_FULL += $(foreach dir,$(INCLUDES),-I$(CURDIR)/$(dir)) \
|
||||||
|
$(foreach dir,$(LIBDIRS),-I$(dir)/include) \
|
||||||
|
$(EXTERNAL_INCLUDE)
|
||||||
|
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
# build a list of library paths
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
export LIBPATHS_FULL += $(foreach dir,$(LIBDIRS),-L$(dir)/lib) \
|
||||||
|
$(EXTERNAL_LIBPATHS)
|
||||||
|
|
||||||
|
|
||||||
|
export OUTPUT := $(CURDIR)/$(TARGET)
|
||||||
|
.PHONY: $(BUILD) clean install
|
||||||
|
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
$(BUILD):
|
||||||
|
@[ -d $@ ] || mkdir -p $@
|
||||||
|
@$(MAKE) --no-print-directory -C $(BUILD) -f $(CURDIR)/Makefile
|
||||||
|
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
clean:
|
||||||
|
@echo clean ...
|
||||||
|
@rm -fr $(BUILD) $(OUTPUT).mod $(OUTPUT)
|
||||||
|
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
else
|
||||||
|
|
||||||
|
DEPENDS := $(OFILES:.o=.d)
|
||||||
|
|
||||||
|
THIS_DIR := $(dir $(abspath $(lastword $(MAKEFILE_LIST))))
|
||||||
|
|
||||||
|
###############################################################################
|
||||||
|
# Rule to make everything.
|
||||||
|
PHONY += all
|
||||||
|
|
||||||
|
all : $(OUTPUT)
|
||||||
|
###############################################################################
|
||||||
|
# Special build rules
|
||||||
|
|
||||||
|
|
||||||
|
# Rule to make the module file.
|
||||||
|
$(OUTPUT) : $(OFILES)
|
||||||
|
@echo "linking ... " $@
|
||||||
|
@$(REAL_LD) $(OFILES) $(LDFLAGS) $(LIBS) $(LIBPATHS_FULL) -o $@
|
||||||
|
|
||||||
|
###############################################################################
|
||||||
|
# Standard build rules
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
%.a:
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
@echo $(notdir $@)
|
||||||
|
@rm -f $@
|
||||||
|
@$(AR) -rc $@ $^
|
||||||
|
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
%.o: %.cpp
|
||||||
|
@echo $(notdir $<)
|
||||||
|
@$(CXX) -MMD -MP -MF $(DEPSDIR)/$*.d $(CXXFLAGS) $(INCLUDE_FULL) -c $< -o $@ $(ERROR_FILTER)
|
||||||
|
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
%.o: %.c
|
||||||
|
@echo $(notdir $<)
|
||||||
|
@$(CC) -MMD -MP -MF $(DEPSDIR)/$*.d $(CFLAGS) $(INCLUDE_FULL) -c $< -o $@ $(ERROR_FILTER)
|
||||||
|
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
%.o: %.S
|
||||||
|
@echo $(notdir $<)
|
||||||
|
$(CC) -MMD -MP -MF $(DEPSDIR)/$*.d -x assembler-with-cpp $(INCLUDE_FULL) -c $< -o $@ $(ERROR_FILTER)
|
||||||
|
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
%.png.o : %.png
|
||||||
|
@echo $(notdir $<)
|
||||||
|
@bin2s -a 32 $< | $(AS) -o $(@)
|
||||||
|
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
%.jpg.o : %.jpg
|
||||||
|
@echo $(notdir $<)
|
||||||
|
@bin2s -a 32 $< | $(AS) -o $(@)
|
||||||
|
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
%.ttf.o : %.ttf
|
||||||
|
@echo $(notdir $<)
|
||||||
|
@bin2s -a 32 $< | $(AS) -o $(@)
|
||||||
|
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
%.bin.o : %.bin
|
||||||
|
@echo $(notdir $<)
|
||||||
|
@bin2s -a 32 $< | $(AS) -o $(@)
|
||||||
|
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
%.wav.o : %.wav
|
||||||
|
@echo $(notdir $<)
|
||||||
|
@bin2s -a 32 $< | $(AS) -o $(@)
|
||||||
|
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
%.mp3.o : %.mp3
|
||||||
|
@echo $(notdir $<)
|
||||||
|
@bin2s -a 32 $< | $(AS) -o $(@)
|
||||||
|
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
%.ogg.o : %.ogg
|
||||||
|
@echo $(notdir $<)
|
||||||
|
@bin2s -a 32 $< | $(AS) -o $(@)
|
||||||
|
|
||||||
|
###############################################################################
|
||||||
|
# Assembly listing rules
|
||||||
|
|
||||||
|
# Rule to make assembly listing.
|
||||||
|
PHONY += list
|
||||||
|
list : $(LIST)
|
||||||
|
|
||||||
|
# Rule to make the listing file.
|
||||||
|
%.list : $(TARGET)
|
||||||
|
$(LOG)
|
||||||
|
-$Qmkdir -p $(dir $@)
|
||||||
|
$Q$(OBJDUMP) -d $< > $@
|
||||||
|
|
||||||
|
###############################################################################
|
||||||
|
# Clean rule
|
||||||
|
|
||||||
|
# Rule to clean files.
|
||||||
|
PHONY += clean
|
||||||
|
clean :
|
||||||
|
$Qrm -rf $(wildcard $(BUILD) $(BIN))
|
||||||
|
|
||||||
|
###############################################################################
|
||||||
|
# Phony targets
|
||||||
|
|
||||||
|
.PHONY : $(PHONY)
|
||||||
|
|
||||||
|
-include $(DEPENDS)
|
||||||
|
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
endif
|
||||||
|
#---------------------------------------------------------------------------------
|
50
makefile.mk
Normal file
50
makefile.mk
Normal file
@ -0,0 +1,50 @@
|
|||||||
|
# Compiling the projects with libutils logging code?
|
||||||
|
DO_LOGGING := 1
|
||||||
|
|
||||||
|
# Target filename
|
||||||
|
TARGET := $(notdir $(CURDIR)).mod
|
||||||
|
|
||||||
|
# Source directories
|
||||||
|
SOURCES := src src/utils
|
||||||
|
|
||||||
|
# Data directories
|
||||||
|
DATA :=
|
||||||
|
|
||||||
|
# Include directories
|
||||||
|
INCLUDES := src
|
||||||
|
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
# options for code generation and linking
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
# Extra C AND C++ compiler flags
|
||||||
|
COMMON_CFLAGS :=
|
||||||
|
# Extra C compiler flags
|
||||||
|
CFLAGS :=
|
||||||
|
# Extra C++ compiler flags
|
||||||
|
CXXFLAGS :=
|
||||||
|
# Extra linking flags for all linking steps
|
||||||
|
LDFLAGS :=
|
||||||
|
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
# list of directories containing libraries, this must be the top level containing
|
||||||
|
# include and lib
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
LIBDIRS := $(WUPSDIR) $(WUT_ROOT) $(PORTLIBS)
|
||||||
|
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
# any extra libraries we wish to link with the project
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
LIBS := -lwups -lwut -liosuhax
|
||||||
|
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
# Will be added to the final lib paths
|
||||||
|
# example:
|
||||||
|
# -L$C:/library1/lib
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
EXTERNAL_LIBPATHS :=
|
||||||
|
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
# Will be added to the final include paths
|
||||||
|
# -IC:/library1/include
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
EXTERNAL_INCLUDE :=
|
53
src/BackgroundThread.cpp
Normal file
53
src/BackgroundThread.cpp
Normal file
@ -0,0 +1,53 @@
|
|||||||
|
#include "BackgroundThread.hpp"
|
||||||
|
#include <malloc.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <coreinit/cache.h>
|
||||||
|
|
||||||
|
#include <utils/logger.h>
|
||||||
|
|
||||||
|
#include "ftp.h"
|
||||||
|
#include "net.h"
|
||||||
|
|
||||||
|
BackgroundThread * BackgroundThread::instance = NULL;
|
||||||
|
|
||||||
|
BackgroundThread::BackgroundThread(): BackgroundThreadWrapper(BackgroundThread::getPriority()) {
|
||||||
|
DEBUG_FUNCTION_LINE("Create new Server\n");
|
||||||
|
mutex.lock();
|
||||||
|
this->serverSocket = create_server(PORT);
|
||||||
|
DCFlushRange(&(this->serverSocket), 4);
|
||||||
|
mutex.unlock();
|
||||||
|
DEBUG_FUNCTION_LINE("handle %d\n", this->serverSocket);
|
||||||
|
resumeThread();
|
||||||
|
}
|
||||||
|
|
||||||
|
BackgroundThread::~BackgroundThread() {
|
||||||
|
DEBUG_FUNCTION_LINE("Clean up FTP\n");
|
||||||
|
if(this->serverSocket != -1){
|
||||||
|
mutex.lock();
|
||||||
|
cleanup_ftp();
|
||||||
|
network_close(this->serverSocket);
|
||||||
|
mutex.unlock();
|
||||||
|
this->serverSocket = -1;
|
||||||
|
}
|
||||||
|
DEBUG_FUNCTION_LINE("Cleaned up FTP\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
BOOL BackgroundThread::whileLoop() {
|
||||||
|
if(this->serverSocket != -1){
|
||||||
|
mutex.lock();
|
||||||
|
network_down = process_ftp_events(this->serverSocket);
|
||||||
|
mutex.unlock();
|
||||||
|
if(network_down) {
|
||||||
|
DEBUG_FUNCTION_LINE("Network is down %d\n", this->serverSocket);
|
||||||
|
mutex.lock();
|
||||||
|
cleanup_ftp();
|
||||||
|
network_close(this->serverSocket);
|
||||||
|
this->serverSocket = -1;
|
||||||
|
DCFlushRange(&(this->serverSocket), 4);
|
||||||
|
mutex.unlock();
|
||||||
|
}
|
||||||
|
OSSleepTicks(OSMillisecondsToTicks(16));
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
50
src/BackgroundThread.hpp
Normal file
50
src/BackgroundThread.hpp
Normal file
@ -0,0 +1,50 @@
|
|||||||
|
#pragma once
|
||||||
|
#include "utils/BackgroundThreadWrapper.hpp"
|
||||||
|
#include <coreinit/cache.h>
|
||||||
|
#include "utils/logger.h"
|
||||||
|
|
||||||
|
#define PORT 21
|
||||||
|
|
||||||
|
class BackgroundThread: BackgroundThreadWrapper {
|
||||||
|
public:
|
||||||
|
static BackgroundThread *getInstance() {
|
||||||
|
DCFlushRange(&instance, sizeof(instance));
|
||||||
|
ICInvalidateRange(&instance, sizeof(instance));
|
||||||
|
if(instance == NULL) {
|
||||||
|
instance = new BackgroundThread();
|
||||||
|
DCFlushRange(&instance, sizeof(instance));
|
||||||
|
ICInvalidateRange(&instance, sizeof(instance));
|
||||||
|
}
|
||||||
|
return instance;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void destroyInstance() {
|
||||||
|
DCFlushRange(&instance, sizeof(instance));
|
||||||
|
ICInvalidateRange(&instance, sizeof(instance));
|
||||||
|
DEBUG_FUNCTION_LINE("Instance is %08X\n", instance);
|
||||||
|
OSSleepTicks(OSSecondsToTicks(1));
|
||||||
|
if(instance != NULL) {
|
||||||
|
delete instance;
|
||||||
|
instance = NULL;
|
||||||
|
DCFlushRange(&instance, sizeof(instance));
|
||||||
|
ICInvalidateRange(&instance, sizeof(instance));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
BackgroundThread();
|
||||||
|
|
||||||
|
virtual ~BackgroundThread();
|
||||||
|
|
||||||
|
private:
|
||||||
|
static int32_t getPriority() {
|
||||||
|
return 16;
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual BOOL whileLoop();
|
||||||
|
|
||||||
|
static BackgroundThread * instance;
|
||||||
|
|
||||||
|
int serverSocket = -1;
|
||||||
|
int network_down = 0;
|
||||||
|
|
||||||
|
};
|
921
src/ftp.c
Normal file
921
src/ftp.c
Normal file
@ -0,0 +1,921 @@
|
|||||||
|
/*
|
||||||
|
|
||||||
|
ftpii -- an FTP server for the Wii
|
||||||
|
|
||||||
|
Copyright (C) 2008 Joseph Jordan <joe.ftpii@psychlaw.com.au>
|
||||||
|
|
||||||
|
This software is provided 'as-is', without any express or implied warranty.
|
||||||
|
In no event will the authors be held liable for any damages arising from
|
||||||
|
the use of this software.
|
||||||
|
|
||||||
|
Permission is granted to anyone to use this software for any purpose,
|
||||||
|
including commercial applications, and to alter it and redistribute it
|
||||||
|
freely, subject to the following restrictions:
|
||||||
|
|
||||||
|
1.The origin of this software must not be misrepresented; you must not
|
||||||
|
claim that you wrote the original software. If you use this software in a
|
||||||
|
product, an acknowledgment in the product documentation would be
|
||||||
|
appreciated but is not required.
|
||||||
|
|
||||||
|
2.Altered source versions must be plainly marked as such, and must not be
|
||||||
|
misrepresented as being the original software.
|
||||||
|
|
||||||
|
3.This notice may not be removed or altered from any source distribution.
|
||||||
|
|
||||||
|
*/
|
||||||
|
#include <malloc.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <sys/dir.h>
|
||||||
|
#include <coreinit/thread.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <nsysnet/socket.h>
|
||||||
|
#include "main.h"
|
||||||
|
#include "utils/logger.h"
|
||||||
|
|
||||||
|
//! TODO: fix those function
|
||||||
|
#define gettime() OSGetTick()
|
||||||
|
#define errno wiiu_geterrno()
|
||||||
|
|
||||||
|
#include "ftp.h"
|
||||||
|
#include "virtualpath.h"
|
||||||
|
#include "net.h"
|
||||||
|
#include "vrt.h"
|
||||||
|
|
||||||
|
#define UNUSED __attribute__((unused))
|
||||||
|
|
||||||
|
#define FTP_BUFFER_SIZE 1024
|
||||||
|
#define MAX_CLIENTS 5
|
||||||
|
|
||||||
|
static const uint16_t SRC_PORT = 20;
|
||||||
|
static const int32_t EQUIT = 696969;
|
||||||
|
static const char *CRLF = "\r\n";
|
||||||
|
static const uint32_t CRLF_LENGTH = 2;
|
||||||
|
|
||||||
|
static uint8_t num_clients = 0;
|
||||||
|
static uint16_t passive_port = 1024;
|
||||||
|
static char *password = NULL;
|
||||||
|
|
||||||
|
void console_printf(const char *format, ...){
|
||||||
|
}
|
||||||
|
|
||||||
|
typedef int32_t (*data_connection_callback)(int32_t data_socket, void *arg);
|
||||||
|
|
||||||
|
struct client_struct {
|
||||||
|
int32_t socket;
|
||||||
|
char representation_type;
|
||||||
|
int32_t passive_socket;
|
||||||
|
int32_t data_socket;
|
||||||
|
char cwd[MAXPATHLEN];
|
||||||
|
char pending_rename[MAXPATHLEN];
|
||||||
|
off_t restart_marker;
|
||||||
|
struct sockaddr_in address;
|
||||||
|
bool authenticated;
|
||||||
|
char buf[FTP_BUFFER_SIZE];
|
||||||
|
int32_t offset;
|
||||||
|
bool data_connection_connected;
|
||||||
|
data_connection_callback data_callback;
|
||||||
|
void *data_connection_callback_arg;
|
||||||
|
void (*data_connection_cleanup)(void *arg);
|
||||||
|
uint64_t data_connection_timer;
|
||||||
|
};
|
||||||
|
|
||||||
|
typedef struct client_struct client_t;
|
||||||
|
|
||||||
|
static client_t *clients[MAX_CLIENTS] = { NULL };
|
||||||
|
|
||||||
|
void set_ftp_password(char *new_password) {
|
||||||
|
if (password)
|
||||||
|
free(password);
|
||||||
|
if (new_password) {
|
||||||
|
password = malloc(strlen(new_password) + 1);
|
||||||
|
if (!password)
|
||||||
|
return;
|
||||||
|
|
||||||
|
strcpy((char *)password, new_password);
|
||||||
|
} else {
|
||||||
|
password = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool compare_ftp_password(char *password_attempt) {
|
||||||
|
return !password || !strcmp((char *)password, password_attempt);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
TODO: support multi-line reply
|
||||||
|
*/
|
||||||
|
static int32_t write_reply(client_t *client, uint16_t code, char *msg) {
|
||||||
|
uint32_t msglen = 4 + strlen(msg) + CRLF_LENGTH;
|
||||||
|
char * msgbuf = (char *) malloc(msglen + 1);
|
||||||
|
if (msgbuf == NULL)
|
||||||
|
return -ENOMEM;
|
||||||
|
sprintf(msgbuf, "%u %s\r\n", code, msg);
|
||||||
|
console_printf("Wrote reply: %s", msgbuf);
|
||||||
|
int32_t ret = send_exact(client->socket, msgbuf, msglen);
|
||||||
|
free(msgbuf);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void close_passive_socket(client_t *client) {
|
||||||
|
if (client->passive_socket >= 0) {
|
||||||
|
network_close_blocking(client->passive_socket);
|
||||||
|
client->passive_socket = -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
result must be able to hold up to maxsplit+1 null-terminated strings of length strlen(s)
|
||||||
|
returns the number of strings stored in the result array (up to maxsplit+1)
|
||||||
|
*/
|
||||||
|
static uint32_t split(char *s, char sep, uint32_t maxsplit, char *result[]) {
|
||||||
|
uint32_t num_results = 0;
|
||||||
|
uint32_t result_pos = 0;
|
||||||
|
uint32_t trim_pos = 0;
|
||||||
|
bool in_word = false;
|
||||||
|
for (; *s; s++) {
|
||||||
|
if (*s == sep) {
|
||||||
|
if (num_results <= maxsplit) {
|
||||||
|
in_word = false;
|
||||||
|
continue;
|
||||||
|
} else if (!trim_pos) {
|
||||||
|
trim_pos = result_pos;
|
||||||
|
}
|
||||||
|
} else if (trim_pos) {
|
||||||
|
trim_pos = 0;
|
||||||
|
}
|
||||||
|
if (!in_word) {
|
||||||
|
in_word = true;
|
||||||
|
if (num_results <= maxsplit) {
|
||||||
|
num_results++;
|
||||||
|
result_pos = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
result[num_results - 1][result_pos++] = *s;
|
||||||
|
result[num_results - 1][result_pos] = '\0';
|
||||||
|
}
|
||||||
|
if (trim_pos) {
|
||||||
|
result[num_results - 1][trim_pos] = '\0';
|
||||||
|
}
|
||||||
|
uint32_t i = num_results;
|
||||||
|
for (i = num_results; i <= maxsplit; i++) {
|
||||||
|
result[i][0] = '\0';
|
||||||
|
}
|
||||||
|
return num_results;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_USER(client_t *client, char *username UNUSED) {
|
||||||
|
return write_reply(client, 331, "User name okay, need password.");
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_PASS(client_t *client, char *password_attempt) {
|
||||||
|
if (compare_ftp_password(password_attempt)) {
|
||||||
|
client->authenticated = true;
|
||||||
|
return write_reply(client, 230, "User logged in, proceed.");
|
||||||
|
} else {
|
||||||
|
return write_reply(client, 530, "Login incorrect.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_REIN(client_t *client, char *rest UNUSED) {
|
||||||
|
close_passive_socket(client);
|
||||||
|
strcpy(client->cwd, "/");
|
||||||
|
client->representation_type = 'A';
|
||||||
|
client->authenticated = false;
|
||||||
|
return write_reply(client, 220, "Service ready for new user.");
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_QUIT(client_t *client, char *rest UNUSED) {
|
||||||
|
// TODO: dont quit if xfer in progress
|
||||||
|
int32_t result = write_reply(client, 221, "Service closing control connection.");
|
||||||
|
return result < 0 ? result : -EQUIT;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_SYST(client_t *client, char *rest UNUSED) {
|
||||||
|
return write_reply(client, 215, "UNIX Type: L8 Version: ftpii");
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_TYPE(client_t *client, char *rest) {
|
||||||
|
char representation_type[FTP_BUFFER_SIZE], param[FTP_BUFFER_SIZE];
|
||||||
|
char *args[] = { representation_type, param };
|
||||||
|
uint32_t num_args = split(rest, ' ', 1, args);
|
||||||
|
if (num_args == 0) {
|
||||||
|
return write_reply(client, 501, "Syntax error in parameters.");
|
||||||
|
} else if ((!strcasecmp("A", representation_type) && (!*param || !strcasecmp("N", param))) ||
|
||||||
|
(!strcasecmp("I", representation_type) && num_args == 1)) {
|
||||||
|
client->representation_type = *representation_type;
|
||||||
|
} else {
|
||||||
|
return write_reply(client, 501, "Syntax error in parameters.");
|
||||||
|
}
|
||||||
|
char msg[15];
|
||||||
|
snprintf(msg, sizeof(msg), "Type set to %s.", representation_type);
|
||||||
|
return write_reply(client, 200, msg);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_MODE(client_t *client, char *rest) {
|
||||||
|
if (!strcasecmp("S", rest)) {
|
||||||
|
return write_reply(client, 200, "Mode S ok.");
|
||||||
|
} else {
|
||||||
|
return write_reply(client, 501, "Syntax error in parameters.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_PWD(client_t *client, char *rest UNUSED) {
|
||||||
|
char msg[MAXPATHLEN + 24];
|
||||||
|
// TODO: escape double-quotes
|
||||||
|
sprintf(msg, "\"%s\" is current directory.", client->cwd);
|
||||||
|
return write_reply(client, 257, msg);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_CWD(client_t *client, char *path) {
|
||||||
|
int32_t result;
|
||||||
|
if (!vrt_chdir(client->cwd, path)) {
|
||||||
|
result = write_reply(client, 250, "CWD command successful.");
|
||||||
|
} else {
|
||||||
|
result = write_reply(client, 550, strerror(errno));
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_CDUP(client_t *client, char *rest UNUSED) {
|
||||||
|
int32_t result;
|
||||||
|
if (!vrt_chdir(client->cwd, "..")) {
|
||||||
|
result = write_reply(client, 250, "CDUP command successful.");
|
||||||
|
} else {
|
||||||
|
result = write_reply(client, 550, strerror(errno));
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_DELE(client_t *client, char *path) {
|
||||||
|
if (!vrt_unlink(client->cwd, path)) {
|
||||||
|
return write_reply(client, 250, "File or directory removed.");
|
||||||
|
} else {
|
||||||
|
return write_reply(client, 550, strerror(errno));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_MKD(client_t *client, char *path) {
|
||||||
|
if (!*path) {
|
||||||
|
return write_reply(client, 501, "Syntax error in parameters.");
|
||||||
|
}
|
||||||
|
if (!vrt_mkdir(client->cwd, path, 0777)) {
|
||||||
|
char msg[MAXPATHLEN + 21];
|
||||||
|
char abspath[MAXPATHLEN];
|
||||||
|
strcpy(abspath, client->cwd);
|
||||||
|
vrt_chdir(abspath, path); // TODO: error checking
|
||||||
|
// TODO: escape double-quotes
|
||||||
|
sprintf(msg, "\"%s\" directory created.", abspath);
|
||||||
|
return write_reply(client, 257, msg);
|
||||||
|
} else {
|
||||||
|
return write_reply(client, 550, strerror(errno));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_RNFR(client_t *client, char *path) {
|
||||||
|
strcpy(client->pending_rename, path);
|
||||||
|
return write_reply(client, 350, "Ready for RNTO.");
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_RNTO(client_t *client, char *path) {
|
||||||
|
if (!*client->pending_rename) {
|
||||||
|
return write_reply(client, 503, "RNFR required first.");
|
||||||
|
}
|
||||||
|
int32_t result;
|
||||||
|
if (!vrt_rename(client->cwd, client->pending_rename, path)) {
|
||||||
|
result = write_reply(client, 250, "Rename successful.");
|
||||||
|
} else {
|
||||||
|
result = write_reply(client, 550, strerror(
|
||||||
|
|
||||||
|
errno));
|
||||||
|
}
|
||||||
|
*client->pending_rename = '\0';
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_SIZE(client_t *client, char *path) {
|
||||||
|
struct stat st;
|
||||||
|
if (!vrt_stat(client->cwd, path, &st)) {
|
||||||
|
char size_buf[12];
|
||||||
|
sprintf(size_buf, "%llu", st.st_size);
|
||||||
|
return write_reply(client, 213, size_buf);
|
||||||
|
} else {
|
||||||
|
return write_reply(client, 550, strerror(errno));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_PASV(client_t *client, char *rest UNUSED) {
|
||||||
|
close_passive_socket(client);
|
||||||
|
client->passive_socket = network_socket(AF_INET, SOCK_STREAM, IPPROTO_IP);
|
||||||
|
if (client->passive_socket < 0) {
|
||||||
|
return write_reply(client, 520, "Unable to create listening socket.");
|
||||||
|
}
|
||||||
|
set_blocking(client->passive_socket, false);
|
||||||
|
struct sockaddr_in bindAddress;
|
||||||
|
memset(&bindAddress, 0, sizeof(bindAddress));
|
||||||
|
bindAddress.sin_family = AF_INET;
|
||||||
|
bindAddress.sin_port = htons(passive_port++); // XXX: BUG: This will overflow eventually, with interesting results...
|
||||||
|
bindAddress.sin_addr.s_addr = htonl(INADDR_ANY);
|
||||||
|
int32_t result;
|
||||||
|
if ((result = network_bind(client->passive_socket, (struct sockaddr *)&bindAddress, sizeof(bindAddress))) < 0) {
|
||||||
|
close_passive_socket(client);
|
||||||
|
return write_reply(client, 520, "Unable to bind listening socket.");
|
||||||
|
}
|
||||||
|
if ((result = network_listen(client->passive_socket, 1)) < 0) {
|
||||||
|
close_passive_socket(client);
|
||||||
|
return write_reply(client, 520, "Unable to listen on socket.");
|
||||||
|
}
|
||||||
|
char reply[49];
|
||||||
|
uint16_t port = bindAddress.sin_port;
|
||||||
|
uint32_t ip = network_gethostip();
|
||||||
|
struct in_addr addr;
|
||||||
|
addr.s_addr = ip;
|
||||||
|
console_printf("Listening for data connections at %s:%u...\n", inet_ntoa(addr), port);
|
||||||
|
sprintf(reply, "Entering Passive Mode (%u,%u,%u,%u,%u,%u).", (ip >> 24) & 0xff, (ip >> 16) & 0xff, (ip >> 8) & 0xff, ip & 0xff, (port >> 8) & 0xff, port & 0xff);
|
||||||
|
return write_reply(client, 227, reply);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_PORT(client_t *client, char *portspec) {
|
||||||
|
uint32_t h1, h2, h3, h4, p1, p2;
|
||||||
|
if (sscanf(portspec, "%3u,%3u,%3u,%3u,%3u,%3u", &h1, &h2, &h3, &h4, &p1, &p2) < 6) {
|
||||||
|
return write_reply(client, 501, "Syntax error in parameters.");
|
||||||
|
}
|
||||||
|
char addr_str[44];
|
||||||
|
sprintf(addr_str, "%u.%u.%u.%u", h1, h2, h3, h4);
|
||||||
|
struct in_addr sin_addr;
|
||||||
|
if (!inet_aton(addr_str, &sin_addr)) {
|
||||||
|
return write_reply(client, 501, "Syntax error in parameters.");
|
||||||
|
}
|
||||||
|
close_passive_socket(client);
|
||||||
|
uint16_t port = ((p1 &0xff) << 8) | (p2 & 0xff);
|
||||||
|
client->address.sin_addr = sin_addr;
|
||||||
|
client->address.sin_port = htons(port);
|
||||||
|
console_printf("Set client address to %s:%u\n", addr_str, port);
|
||||||
|
return write_reply(client, 200, "PORT command successful.");
|
||||||
|
}
|
||||||
|
|
||||||
|
typedef int32_t (*data_connection_handler)(client_t *client, data_connection_callback callback, void *arg);
|
||||||
|
|
||||||
|
static int32_t prepare_data_connection_active(client_t *client, data_connection_callback callback UNUSED, void *arg UNUSED) {
|
||||||
|
int32_t data_socket = network_socket(AF_INET, SOCK_STREAM, IPPROTO_IP);
|
||||||
|
if (data_socket < 0)
|
||||||
|
return data_socket;
|
||||||
|
set_blocking(data_socket, false);
|
||||||
|
struct sockaddr_in bindAddress;
|
||||||
|
memset(&bindAddress, 0, sizeof(bindAddress));
|
||||||
|
bindAddress.sin_family = AF_INET;
|
||||||
|
bindAddress.sin_port = htons(SRC_PORT);
|
||||||
|
bindAddress.sin_addr.s_addr = htonl(INADDR_ANY);
|
||||||
|
int32_t result;
|
||||||
|
if ((result = network_bind(data_socket, (struct sockaddr *)&bindAddress, sizeof(bindAddress))) < 0) {
|
||||||
|
network_close(data_socket);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
client->data_socket = data_socket;
|
||||||
|
console_printf("Attempting to connect to client at %s:%u\n", inet_ntoa(client->address.sin_addr), client->address.sin_port);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t prepare_data_connection_passive(client_t *client, data_connection_callback callback UNUSED, void *arg UNUSED) {
|
||||||
|
client->data_socket = client->passive_socket;
|
||||||
|
console_printf("Waiting for data connections...\n");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t prepare_data_connection(client_t *client, void *callback, void *arg, void *cleanup) {
|
||||||
|
int32_t result = write_reply(client, 150, "Transferring data.");
|
||||||
|
if (result >= 0) {
|
||||||
|
data_connection_handler handler = prepare_data_connection_active;
|
||||||
|
if (client->passive_socket >= 0)
|
||||||
|
handler = prepare_data_connection_passive;
|
||||||
|
result = handler(client, (data_connection_callback)callback, arg);
|
||||||
|
if (result < 0) {
|
||||||
|
result = write_reply(client, 520, "Closing data connection, error occurred during transfer.");
|
||||||
|
} else {
|
||||||
|
client->data_connection_connected = false;
|
||||||
|
client->data_callback = callback;
|
||||||
|
client->data_connection_callback_arg = arg;
|
||||||
|
client->data_connection_cleanup = cleanup;
|
||||||
|
client->data_connection_timer = gettime() + OSSecondsToTicks(10);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t send_nlst(int32_t data_socket, DIR_P *iter) {
|
||||||
|
int32_t result = 0;
|
||||||
|
char filename[MAXPATHLEN];
|
||||||
|
struct dirent *dirent = NULL;
|
||||||
|
while ((dirent = vrt_readdir(iter)) != 0) {
|
||||||
|
size_t end_index = strlen(dirent->d_name);
|
||||||
|
if(end_index + 2 >= MAXPATHLEN)
|
||||||
|
continue;
|
||||||
|
strcpy(filename, dirent->d_name);
|
||||||
|
filename[end_index] = CRLF[0];
|
||||||
|
filename[end_index + 1] = CRLF[1];
|
||||||
|
filename[end_index + 2] = '\0';
|
||||||
|
if ((result = send_exact(data_socket, filename, strlen(filename))) < 0) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return result < 0 ? result : 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t send_list(int32_t data_socket, DIR_P *iter) {
|
||||||
|
struct stat st;
|
||||||
|
int32_t result = 0;
|
||||||
|
time_t mtime = 0;
|
||||||
|
uint64_t size = 0;
|
||||||
|
char filename[MAXPATHLEN];
|
||||||
|
char line[MAXPATHLEN + 56 + CRLF_LENGTH + 1];
|
||||||
|
struct dirent *dirent = NULL;
|
||||||
|
while ((dirent = vrt_readdir(iter)) != 0) {
|
||||||
|
|
||||||
|
snprintf(filename, sizeof(filename), "%s/%s", iter->path, dirent->d_name);
|
||||||
|
if(stat(filename, &st) == 0) {
|
||||||
|
mtime = st.st_mtime;
|
||||||
|
size = st.st_size;
|
||||||
|
} else {
|
||||||
|
mtime = time(0);
|
||||||
|
size = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
char timestamp[13];
|
||||||
|
strftime(timestamp, sizeof(timestamp), "%b %d %Y", localtime(&mtime));
|
||||||
|
snprintf(line, sizeof(line), "%crwxr-xr-x 1 0 0 %10llu %s %s\r\n", (dirent->d_type & DT_DIR) ? 'd' : '-', size, timestamp, dirent->d_name);
|
||||||
|
if ((result = send_exact(data_socket, line, strlen(line))) < 0) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return result < 0 ? result : 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_NLST(client_t *client, char *path) {
|
||||||
|
if (!*path) {
|
||||||
|
path = ".";
|
||||||
|
}
|
||||||
|
|
||||||
|
DIR_P *dir = vrt_opendir(client->cwd, path);
|
||||||
|
if (dir == NULL) {
|
||||||
|
return write_reply(client, 550, strerror(errno));
|
||||||
|
}
|
||||||
|
|
||||||
|
int32_t result = prepare_data_connection(client, send_nlst, dir, vrt_closedir);
|
||||||
|
if (result < 0)
|
||||||
|
vrt_closedir(dir);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_LIST(client_t *client, char *path) {
|
||||||
|
if (*path == '-') {
|
||||||
|
// handle buggy clients that use "LIST -aL" or similar, at the expense of breaking paths that begin with '-'
|
||||||
|
char flags[FTP_BUFFER_SIZE];
|
||||||
|
char rest[FTP_BUFFER_SIZE];
|
||||||
|
char *args[] = { flags, rest };
|
||||||
|
split(path, ' ', 1, args);
|
||||||
|
path = rest;
|
||||||
|
}
|
||||||
|
if (!*path) {
|
||||||
|
path = ".";
|
||||||
|
}
|
||||||
|
|
||||||
|
if(path && client->cwd) {
|
||||||
|
if(strcmp(path, ".") == 0 && strcmp(client->cwd, "/") == 0) {
|
||||||
|
UnmountVirtualPaths();
|
||||||
|
MountVirtualDevices();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
DIR_P *dir = vrt_opendir(client->cwd, path);
|
||||||
|
if (dir == NULL) {
|
||||||
|
return write_reply(client, 550, strerror(errno));
|
||||||
|
}
|
||||||
|
|
||||||
|
int32_t result = prepare_data_connection(client, send_list, dir, vrt_closedir);
|
||||||
|
if (result < 0)
|
||||||
|
vrt_closedir(dir);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_RETR(client_t *client, char *path) {
|
||||||
|
FILE *f = vrt_fopen(client->cwd, path, "rb");
|
||||||
|
if (!f) {
|
||||||
|
return write_reply(client, 550, strerror(errno));
|
||||||
|
}
|
||||||
|
|
||||||
|
int fd = fileno(f);
|
||||||
|
if (client->restart_marker && lseek(fd, client->restart_marker, SEEK_SET) != client->restart_marker) {
|
||||||
|
int32_t lseek_error = errno;
|
||||||
|
fclose(f);
|
||||||
|
client->restart_marker = 0;
|
||||||
|
return write_reply(client, 550, strerror(lseek_error));
|
||||||
|
}
|
||||||
|
client->restart_marker = 0;
|
||||||
|
|
||||||
|
int32_t result = prepare_data_connection(client, send_from_file, f, fclose);
|
||||||
|
if (result < 0)
|
||||||
|
fclose(f);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t stor_or_append(client_t *client, FILE *f) {
|
||||||
|
if (!f) {
|
||||||
|
return write_reply(client, 550, strerror(errno));
|
||||||
|
}
|
||||||
|
int32_t result = prepare_data_connection(client, recv_to_file, f, fclose);
|
||||||
|
if (result < 0)
|
||||||
|
fclose(f);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_STOR(client_t *client, char *path) {
|
||||||
|
FILE *f = vrt_fopen(client->cwd, path, "wb");
|
||||||
|
int fd;
|
||||||
|
if (f)
|
||||||
|
fd = fileno(f);
|
||||||
|
if (f && client->restart_marker && lseek(fd, client->restart_marker, SEEK_SET) != client->restart_marker) {
|
||||||
|
int32_t lseek_error = errno;
|
||||||
|
fclose(f);
|
||||||
|
client->restart_marker = 0;
|
||||||
|
return write_reply(client, 550, strerror(lseek_error));
|
||||||
|
}
|
||||||
|
client->restart_marker = 0;
|
||||||
|
|
||||||
|
return stor_or_append(client, f);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_APPE(client_t *client, char *path) {
|
||||||
|
return stor_or_append(client, vrt_fopen(client->cwd, path, "ab"));
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_REST(client_t *client, char *offset_str) {
|
||||||
|
off_t offset;
|
||||||
|
if (sscanf(offset_str, "%lli", &offset) < 1 || offset < 0) {
|
||||||
|
return write_reply(client, 501, "Syntax error in parameters.");
|
||||||
|
}
|
||||||
|
client->restart_marker = offset;
|
||||||
|
char msg[FTP_BUFFER_SIZE];
|
||||||
|
sprintf(msg, "Restart position accepted (%lli).", offset);
|
||||||
|
return write_reply(client, 350, msg);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_SITE_LOADER(client_t *client, char *rest UNUSED) {
|
||||||
|
int32_t result = write_reply(client, 200, "Exiting to loader.");
|
||||||
|
//set_reset_flag();
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_SITE_CLEAR(client_t *client, char *rest UNUSED) {
|
||||||
|
int32_t result = write_reply(client, 200, "Cleared.");
|
||||||
|
uint32_t i;
|
||||||
|
for (i = 0; i < 18; i++)
|
||||||
|
console_printf("\n");
|
||||||
|
//console_printf("\x1b[2;0H");
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
This is implemented as a no-op to prevent some FTP clients
|
||||||
|
from displaying skip/abort/retry type prompts.
|
||||||
|
*/
|
||||||
|
static int32_t ftp_SITE_CHMOD(client_t *client, char *rest UNUSED) {
|
||||||
|
return write_reply(client, 250, "SITE CHMOD command ok.");
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_SITE_PASSWD(client_t *client, char *new_password) {
|
||||||
|
set_ftp_password(new_password);
|
||||||
|
return write_reply(client, 200, "Password changed.");
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_SITE_NOPASSWD(client_t *client, char *rest UNUSED) {
|
||||||
|
set_ftp_password(NULL);
|
||||||
|
return write_reply(client, 200, "Authentication disabled.");
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_SITE_EJECT(client_t *client, char *rest UNUSED) {
|
||||||
|
//if (dvd_eject()) return write_reply(client, 550, "Unable to eject DVD.");
|
||||||
|
return write_reply(client, 200, "DVD ejected.");
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_SITE_MOUNT(client_t *client, char *path UNUSED) {
|
||||||
|
//if (!mount_virtual(path)) return write_reply(client, 550, "Unable to mount.");
|
||||||
|
return write_reply(client, 250, "Mounted.");
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_SITE_UNMOUNT(client_t *client, char *path UNUSED) {
|
||||||
|
//if (!unmount_virtual(path)) return write_reply(client, 550, "Unable to unmount.");
|
||||||
|
return write_reply(client, 250, "Unmounted.");
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_SITE_UNKNOWN(client_t *client, char *rest UNUSED) {
|
||||||
|
return write_reply(client, 501, "Unknown SITE command.");
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_SITE_LOAD(client_t *client, char *path UNUSED) {
|
||||||
|
// FILE *f = vrt_fopen(client->cwd, path, "rb");
|
||||||
|
// if (!f) return write_reply(client, 550, strerror(errno));
|
||||||
|
// char *real_path = to_real_path(client->cwd, path);
|
||||||
|
// if (!real_path) goto end;
|
||||||
|
// load_from_file(f, real_path);
|
||||||
|
// free(real_path);
|
||||||
|
// end:
|
||||||
|
// fclose(f);
|
||||||
|
return write_reply(client, 500, "Unable to load.");
|
||||||
|
}
|
||||||
|
|
||||||
|
typedef int32_t (*ftp_command_handler)(client_t *client, char *args);
|
||||||
|
|
||||||
|
static int32_t dispatch_to_handler(client_t *client, char *cmd_line, const char **commands, const ftp_command_handler *handlers) {
|
||||||
|
char cmd[FTP_BUFFER_SIZE], rest[FTP_BUFFER_SIZE];
|
||||||
|
char *args[] = { cmd, rest };
|
||||||
|
split(cmd_line, ' ', 1, args);
|
||||||
|
int32_t i;
|
||||||
|
for (i = 0; commands[i]; i++) {
|
||||||
|
if (!strcasecmp(commands[i], cmd))
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return handlers[i](client, rest);
|
||||||
|
}
|
||||||
|
|
||||||
|
static const char *site_commands[] = { "LOADER", "CLEAR", "CHMOD", "PASSWD", "NOPASSWD", "EJECT", "MOUNT", "UNMOUNT", "LOAD", NULL };
|
||||||
|
static const ftp_command_handler site_handlers[] = { ftp_SITE_LOADER, ftp_SITE_CLEAR, ftp_SITE_CHMOD, ftp_SITE_PASSWD, ftp_SITE_NOPASSWD, ftp_SITE_EJECT, ftp_SITE_MOUNT, ftp_SITE_UNMOUNT, ftp_SITE_LOAD, ftp_SITE_UNKNOWN };
|
||||||
|
|
||||||
|
static int32_t ftp_SITE(client_t *client, char *cmd_line) {
|
||||||
|
return dispatch_to_handler(client, cmd_line, site_commands, site_handlers);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_NOOP(client_t *client, char *rest UNUSED) {
|
||||||
|
return write_reply(client, 200, "NOOP command successful.");
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_SUPERFLUOUS(client_t *client, char *rest UNUSED) {
|
||||||
|
return write_reply(client, 202, "Command not implemented, superfluous at this site.");
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_NEEDAUTH(client_t *client, char *rest UNUSED) {
|
||||||
|
return write_reply(client, 530, "Please login with USER and PASS.");
|
||||||
|
}
|
||||||
|
|
||||||
|
static int32_t ftp_UNKNOWN(client_t *client, char *rest UNUSED) {
|
||||||
|
return write_reply(client, 502, "Command not implemented.");
|
||||||
|
}
|
||||||
|
|
||||||
|
static const char *unauthenticated_commands[] = { "USER", "PASS", "QUIT", "REIN", "NOOP", NULL };
|
||||||
|
static const ftp_command_handler unauthenticated_handlers[] = { ftp_USER, ftp_PASS, ftp_QUIT, ftp_REIN, ftp_NOOP, ftp_NEEDAUTH };
|
||||||
|
|
||||||
|
static const char *authenticated_commands[] = {
|
||||||
|
"USER", "PASS", "LIST", "PWD", "CWD", "CDUP",
|
||||||
|
"SIZE", "PASV", "PORT", "TYPE", "SYST", "MODE",
|
||||||
|
"RETR", "STOR", "APPE", "REST", "DELE", "MKD",
|
||||||
|
"RMD", "RNFR", "RNTO", "NLST", "QUIT", "REIN",
|
||||||
|
"SITE", "NOOP", "ALLO", NULL
|
||||||
|
};
|
||||||
|
static const ftp_command_handler authenticated_handlers[] = {
|
||||||
|
ftp_USER, ftp_PASS, ftp_LIST, ftp_PWD, ftp_CWD, ftp_CDUP,
|
||||||
|
ftp_SIZE, ftp_PASV, ftp_PORT, ftp_TYPE, ftp_SYST, ftp_MODE,
|
||||||
|
ftp_RETR, ftp_STOR, ftp_APPE, ftp_REST, ftp_DELE, ftp_MKD,
|
||||||
|
ftp_DELE, ftp_RNFR, ftp_RNTO, ftp_NLST, ftp_QUIT, ftp_REIN,
|
||||||
|
ftp_SITE, ftp_NOOP, ftp_SUPERFLUOUS, ftp_UNKNOWN
|
||||||
|
};
|
||||||
|
|
||||||
|
/*
|
||||||
|
returns negative to signal an error that requires closing the connection
|
||||||
|
*/
|
||||||
|
static int32_t process_command(client_t *client, char *cmd_line) {
|
||||||
|
if (strlen(cmd_line) == 0) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
console_printf("Got command: %s\n", cmd_line);
|
||||||
|
|
||||||
|
const char **commands = unauthenticated_commands;
|
||||||
|
const ftp_command_handler *handlers = unauthenticated_handlers;
|
||||||
|
|
||||||
|
if (client->authenticated) {
|
||||||
|
commands = authenticated_commands;
|
||||||
|
handlers = authenticated_handlers;
|
||||||
|
}
|
||||||
|
|
||||||
|
return dispatch_to_handler(client, cmd_line, commands, handlers);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void cleanup_data_resources(client_t *client) {
|
||||||
|
if (client->data_socket >= 0 && client->data_socket != client->passive_socket) {
|
||||||
|
network_close_blocking(client->data_socket);
|
||||||
|
}
|
||||||
|
client->data_socket = -1;
|
||||||
|
client->data_connection_connected = false;
|
||||||
|
client->data_callback = NULL;
|
||||||
|
if (client->data_connection_cleanup) {
|
||||||
|
client->data_connection_cleanup(client->data_connection_callback_arg);
|
||||||
|
}
|
||||||
|
client->data_connection_callback_arg = NULL;
|
||||||
|
client->data_connection_cleanup = NULL;
|
||||||
|
client->data_connection_timer = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void cleanup_client(client_t *client) {
|
||||||
|
network_close_blocking(client->socket);
|
||||||
|
cleanup_data_resources(client);
|
||||||
|
close_passive_socket(client);
|
||||||
|
int client_index;
|
||||||
|
for (client_index = 0; client_index < MAX_CLIENTS; client_index++) {
|
||||||
|
if (clients[client_index] == client) {
|
||||||
|
clients[client_index] = NULL;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
free(client);
|
||||||
|
num_clients--;
|
||||||
|
console_printf("Client disconnected.\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
void cleanup_ftp() {
|
||||||
|
int client_index;
|
||||||
|
for (client_index = 0; client_index < MAX_CLIENTS; client_index++) {
|
||||||
|
client_t *client = clients[client_index];
|
||||||
|
if (client) {
|
||||||
|
write_reply(client, 421, "Service not available, closing control connection.");
|
||||||
|
cleanup_client(client);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool process_accept_events(int32_t server) {
|
||||||
|
int32_t peer;
|
||||||
|
struct sockaddr_in client_address;
|
||||||
|
int32_t addrlen = sizeof(client_address);
|
||||||
|
while ((peer = network_accept(server, (struct sockaddr *)&client_address, &addrlen)) != -WIIU_EAGAIN) {
|
||||||
|
if (peer < 0) {
|
||||||
|
console_printf("Error accepting connection: [%i] %s\n", -peer, strerror(-peer));
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
console_printf("Accepted connection from %s!\n", inet_ntoa(client_address.sin_addr));
|
||||||
|
|
||||||
|
if (num_clients == MAX_CLIENTS) {
|
||||||
|
console_printf("Maximum of %u clients reached, not accepting client.\n", MAX_CLIENTS);
|
||||||
|
network_close(peer);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
client_t *client = malloc(sizeof(client_t));
|
||||||
|
if (!client) {
|
||||||
|
console_printf("Could not allocate memory for client state, not accepting client.\n");
|
||||||
|
network_close(peer);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
client->socket = peer;
|
||||||
|
client->representation_type = 'A';
|
||||||
|
client->passive_socket = -1;
|
||||||
|
client->data_socket = -1;
|
||||||
|
strcpy(client->cwd, "/");
|
||||||
|
*client->pending_rename = '\0';
|
||||||
|
client->restart_marker = 0;
|
||||||
|
client->authenticated = false;
|
||||||
|
client->offset = 0;
|
||||||
|
client->data_connection_connected = false;
|
||||||
|
client->data_callback = NULL;
|
||||||
|
client->data_connection_callback_arg = NULL;
|
||||||
|
client->data_connection_cleanup = NULL;
|
||||||
|
client->data_connection_timer = 0;
|
||||||
|
memcpy(&client->address, &client_address, sizeof(client_address));
|
||||||
|
int client_index;
|
||||||
|
if (write_reply(client, 220, "ftpii") < 0) {
|
||||||
|
console_printf("Error writing greeting.\n");
|
||||||
|
network_close_blocking(peer);
|
||||||
|
free(client);
|
||||||
|
} else {
|
||||||
|
for (client_index = 0; client_index < MAX_CLIENTS; client_index++) {
|
||||||
|
if (!clients[client_index]) {
|
||||||
|
clients[client_index] = client;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
num_clients++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void process_data_events(client_t *client) {
|
||||||
|
int32_t result;
|
||||||
|
if (!client->data_connection_connected) {
|
||||||
|
if (client->passive_socket >= 0) {
|
||||||
|
struct sockaddr_in data_peer_address;
|
||||||
|
int32_t addrlen = sizeof(data_peer_address);
|
||||||
|
result = network_accept(client->passive_socket, (struct sockaddr *)&data_peer_address,&addrlen);
|
||||||
|
if (result >= 0) {
|
||||||
|
client->data_socket = result;
|
||||||
|
client->data_connection_connected = true;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if ((result = network_connect(client->data_socket, (struct sockaddr *)&client->address, sizeof(client->address))) < 0) {
|
||||||
|
if (result == -EINPROGRESS || result == -EALREADY)
|
||||||
|
result = -WIIU_EAGAIN;
|
||||||
|
if ((result != -WIIU_EAGAIN) && (result != -EISCONN)) {
|
||||||
|
console_printf("Unable to connect to client: [%i] %s\n", -result, strerror(-result));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (result >= 0 || result == -EISCONN) {
|
||||||
|
client->data_connection_connected = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (client->data_connection_connected) {
|
||||||
|
result = 1;
|
||||||
|
console_printf("Connected to client! Transferring data...\n");
|
||||||
|
} else if (gettime() > client->data_connection_timer) {
|
||||||
|
result = -2;
|
||||||
|
console_printf("Timed out waiting for data connection.\n");
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
result = client->data_callback(client->data_socket, client->data_connection_callback_arg);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (result <= 0 && result != -WIIU_EAGAIN) {
|
||||||
|
cleanup_data_resources(client);
|
||||||
|
if (result < 0) {
|
||||||
|
result = write_reply(client, 520, "Closing data connection, error occurred during transfer.");
|
||||||
|
} else {
|
||||||
|
result = write_reply(client, 226, "Closing data connection, transfer successful.");
|
||||||
|
}
|
||||||
|
if (result < 0) {
|
||||||
|
cleanup_client(client);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void process_control_events(client_t *client) {
|
||||||
|
int32_t bytes_read;
|
||||||
|
while (client->offset < (FTP_BUFFER_SIZE - 1)) {
|
||||||
|
if (client->data_callback) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
char *offset_buf = client->buf + client->offset;
|
||||||
|
if ((bytes_read = network_read(client->socket, offset_buf, FTP_BUFFER_SIZE - 1 - client->offset)) < 0) {
|
||||||
|
if (bytes_read != -EAGAIN) {
|
||||||
|
console_printf("Read error %i occurred, closing client.\n", bytes_read);
|
||||||
|
goto recv_loop_end;
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
} else if (bytes_read == 0) {
|
||||||
|
goto recv_loop_end; // EOF from client
|
||||||
|
}
|
||||||
|
client->offset += bytes_read;
|
||||||
|
client->buf[client->offset] = '\0';
|
||||||
|
|
||||||
|
if (strchr(offset_buf, '\0') != (client->buf + client->offset)) {
|
||||||
|
console_printf("Received a null byte from client, closing connection ;-)\n"); // i have decided this isn't allowed =P
|
||||||
|
goto recv_loop_end;
|
||||||
|
}
|
||||||
|
|
||||||
|
char *next;
|
||||||
|
char *end;
|
||||||
|
for (next = client->buf; (end = strstr(next, CRLF)) && !client->data_callback; next = end + CRLF_LENGTH) {
|
||||||
|
*end = '\0';
|
||||||
|
if (strchr(next, '\n')) {
|
||||||
|
console_printf("Received a line-feed from client without preceding carriage return, closing connection ;-)\n"); // i have decided this isn't allowed =P
|
||||||
|
goto recv_loop_end;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (*next) {
|
||||||
|
int32_t result;
|
||||||
|
if ((result = process_command(client, next)) < 0) {
|
||||||
|
if (result != -EQUIT) {
|
||||||
|
console_printf("Closing connection due to error while processing command: %s\n", next);
|
||||||
|
}
|
||||||
|
goto recv_loop_end;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
if (next != client->buf) { // some lines were processed
|
||||||
|
client->offset = strlen(next);
|
||||||
|
char tmp_buf[client->offset];
|
||||||
|
memcpy(tmp_buf, next, client->offset);
|
||||||
|
memcpy(client->buf, tmp_buf, client->offset);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
console_printf("Received line longer than %u bytes, closing client.\n", FTP_BUFFER_SIZE - 1);
|
||||||
|
|
||||||
|
recv_loop_end:
|
||||||
|
cleanup_client(client);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool process_ftp_events(int32_t server) {
|
||||||
|
bool network_down = !process_accept_events(server);
|
||||||
|
int client_index;
|
||||||
|
for (client_index = 0; client_index < MAX_CLIENTS; client_index++) {
|
||||||
|
client_t *client = clients[client_index];
|
||||||
|
if (client) {
|
||||||
|
if (client->data_callback) {
|
||||||
|
process_data_events(client);
|
||||||
|
} else {
|
||||||
|
process_control_events(client);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return network_down;
|
||||||
|
}
|
43
src/ftp.h
Normal file
43
src/ftp.h
Normal file
@ -0,0 +1,43 @@
|
|||||||
|
/*
|
||||||
|
|
||||||
|
ftpii -- an FTP server for the Wii
|
||||||
|
|
||||||
|
Copyright (C) 2008 Joseph Jordan <joe.ftpii@psychlaw.com.au>
|
||||||
|
|
||||||
|
This software is provided 'as-is', without any express or implied warranty.
|
||||||
|
In no event will the authors be held liable for any damages arising from
|
||||||
|
the use of this software.
|
||||||
|
|
||||||
|
Permission is granted to anyone to use this software for any purpose,
|
||||||
|
including commercial applications, and to alter it and redistribute it
|
||||||
|
freely, subject to the following restrictions:
|
||||||
|
|
||||||
|
1.The origin of this software must not be misrepresented; you must not
|
||||||
|
claim that you wrote the original software. If you use this software in a
|
||||||
|
product, an acknowledgment in the product documentation would be
|
||||||
|
appreciated but is not required.
|
||||||
|
|
||||||
|
2.Altered source versions must be plainly marked as such, and must not be
|
||||||
|
misrepresented as being the original software.
|
||||||
|
|
||||||
|
3.This notice may not be removed or altered from any source distribution.
|
||||||
|
|
||||||
|
*/
|
||||||
|
#ifndef _FTP_H_
|
||||||
|
#define _FTP_H_
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C"{
|
||||||
|
#endif
|
||||||
|
#include <stdbool.h>
|
||||||
|
|
||||||
|
void accept_ftp_client(int32_t server);
|
||||||
|
void set_ftp_password(char *new_password);
|
||||||
|
bool process_ftp_events(int32_t server);
|
||||||
|
void cleanup_ftp();
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif /* _FTP_H_ */
|
138
src/main.cpp
Normal file
138
src/main.cpp
Normal file
@ -0,0 +1,138 @@
|
|||||||
|
|
||||||
|
#include <wups.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <stdarg.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <malloc.h>
|
||||||
|
#include <stdint.h>
|
||||||
|
#include <iosuhax.h>
|
||||||
|
#include <iosuhax_devoptab.h>
|
||||||
|
#include <iosuhax_disc_interface.h>
|
||||||
|
#include <proc_ui/procui.h>
|
||||||
|
#include <coreinit/foreground.h>
|
||||||
|
|
||||||
|
#include <coreinit/screen.h>
|
||||||
|
#include <sysapp/launch.h>
|
||||||
|
#include <coreinit/dynload.h>
|
||||||
|
#include <nn/ac.h>
|
||||||
|
#include <stdarg.h>
|
||||||
|
#include <whb/proc.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <coreinit/thread.h>
|
||||||
|
#include <coreinit/cache.h>
|
||||||
|
#include <coreinit/time.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <whb/libmanager.h>
|
||||||
|
#include "utils/logger.h"
|
||||||
|
#include "utils/utils.h"
|
||||||
|
|
||||||
|
#include "virtualpath.h"
|
||||||
|
#include "net.h"
|
||||||
|
#include "BackgroundThread.hpp"
|
||||||
|
|
||||||
|
#define MAX_CONSOLE_LINES_TV 27
|
||||||
|
#define MAX_CONSOLE_LINES_DRC 18
|
||||||
|
|
||||||
|
WUPS_PLUGIN_NAME("FTPiiU");
|
||||||
|
WUPS_PLUGIN_DESCRIPTION("FTP Server");
|
||||||
|
WUPS_PLUGIN_VERSION("0.1");
|
||||||
|
WUPS_PLUGIN_AUTHOR("Maschell");
|
||||||
|
WUPS_PLUGIN_LICENSE("GPL");
|
||||||
|
|
||||||
|
WUPS_USE_WUT_CRT()
|
||||||
|
|
||||||
|
uint32_t hostIpAddress = 0;
|
||||||
|
int iosuhaxMount = 0;
|
||||||
|
int fsaFd = 0;
|
||||||
|
|
||||||
|
BackgroundThread * thread = NULL;
|
||||||
|
|
||||||
|
/* Entry point */
|
||||||
|
ON_APPLICATION_START(args) {
|
||||||
|
WHBInitializeSocketLibrary();
|
||||||
|
|
||||||
|
nn::ac::ConfigIdNum configId;
|
||||||
|
|
||||||
|
nn::ac::Initialize();
|
||||||
|
nn::ac::GetStartupId(&configId);
|
||||||
|
nn::ac::Connect(configId);
|
||||||
|
|
||||||
|
ACGetAssignedAddress(&hostIpAddress);
|
||||||
|
|
||||||
|
log_init();
|
||||||
|
|
||||||
|
//!*******************************************************************
|
||||||
|
//! Initialize FS *
|
||||||
|
//!*******************************************************************
|
||||||
|
|
||||||
|
int fsaFd = -1;
|
||||||
|
|
||||||
|
DEBUG_FUNCTION_LINE("IOSUHAX_Open\n");
|
||||||
|
int res = IOSUHAX_Open(NULL);
|
||||||
|
if(res < 0) {
|
||||||
|
DEBUG_FUNCTION_LINE("IOSUHAX_open failed\n");
|
||||||
|
VirtualMountDevice("fs:/");
|
||||||
|
} else {
|
||||||
|
iosuhaxMount = 1;
|
||||||
|
//fatInitDefault();
|
||||||
|
|
||||||
|
DEBUG_FUNCTION_LINE("IOSUHAX_FSA_Open\n");
|
||||||
|
fsaFd = IOSUHAX_FSA_Open();
|
||||||
|
if(fsaFd < 0) {
|
||||||
|
DEBUG_FUNCTION_LINE("IOSUHAX_FSA_Open failed\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
DEBUG_FUNCTION_LINE("IOSUHAX_FSA_Open\n");
|
||||||
|
|
||||||
|
mount_fs("slccmpt01", fsaFd, "/dev/slccmpt01", "/vol/storage_slccmpt01");
|
||||||
|
mount_fs("storage_odd_tickets", fsaFd, "/dev/odd01", "/vol/storage_odd_tickets");
|
||||||
|
mount_fs("storage_odd_updates", fsaFd, "/dev/odd02", "/vol/storage_odd_updates");
|
||||||
|
mount_fs("storage_odd_content", fsaFd, "/dev/odd03", "/vol/storage_odd_content");
|
||||||
|
mount_fs("storage_odd_content2", fsaFd, "/dev/odd04", "/vol/storage_odd_content2");
|
||||||
|
mount_fs("storage_slc", fsaFd, NULL, "/vol/system");
|
||||||
|
mount_fs("storage_mlc", fsaFd, NULL, "/vol/storage_mlc01");
|
||||||
|
mount_fs("storage_usb", fsaFd, NULL, "/vol/storage_usb01");
|
||||||
|
|
||||||
|
VirtualMountDevice("fs:/");
|
||||||
|
VirtualMountDevice("slccmpt01:/");
|
||||||
|
VirtualMountDevice("storage_odd_tickets:/");
|
||||||
|
VirtualMountDevice("storage_odd_updates:/");
|
||||||
|
VirtualMountDevice("storage_odd_content:/");
|
||||||
|
VirtualMountDevice("storage_odd_content2:/");
|
||||||
|
VirtualMountDevice("storage_slc:/");
|
||||||
|
VirtualMountDevice("storage_mlc:/");
|
||||||
|
VirtualMountDevice("storage_usb:/");
|
||||||
|
VirtualMountDevice("usb:/");
|
||||||
|
}
|
||||||
|
|
||||||
|
thread = BackgroundThread::getInstance();
|
||||||
|
DCFlushRange(&thread, 4);
|
||||||
|
}
|
||||||
|
|
||||||
|
void stopThread(){
|
||||||
|
BackgroundThread::destroyInstance();
|
||||||
|
}
|
||||||
|
|
||||||
|
ON_APPLICATION_END(){
|
||||||
|
DEBUG_FUNCTION_LINE("Ending ftp server\n");
|
||||||
|
stopThread();
|
||||||
|
|
||||||
|
if(iosuhaxMount) {
|
||||||
|
IOSUHAX_sdio_disc_interface.shutdown();
|
||||||
|
IOSUHAX_usb_disc_interface.shutdown();
|
||||||
|
|
||||||
|
unmount_fs("slccmpt01");
|
||||||
|
unmount_fs("storage_odd_tickets");
|
||||||
|
unmount_fs("storage_odd_updates");
|
||||||
|
unmount_fs("storage_odd_content");
|
||||||
|
unmount_fs("storage_odd_content2");
|
||||||
|
unmount_fs("storage_slc");
|
||||||
|
unmount_fs("storage_mlc");
|
||||||
|
unmount_fs("storage_usb");
|
||||||
|
IOSUHAX_FSA_Close(fsaFd);
|
||||||
|
IOSUHAX_Close();
|
||||||
|
}
|
||||||
|
|
||||||
|
UnmountVirtualPaths();
|
||||||
|
}
|
||||||
|
|
32
src/main.h
Normal file
32
src/main.h
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
#ifndef _MAIN_H_
|
||||||
|
#define _MAIN_H_
|
||||||
|
|
||||||
|
/* Main */
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
#include <stdint.h>
|
||||||
|
#include <nsysnet/socket.h>
|
||||||
|
|
||||||
|
#define MAXPATHLEN 256
|
||||||
|
|
||||||
|
#define WIIU_EAGAIN EWOULDBLOCK
|
||||||
|
#define ENODATA 1
|
||||||
|
#define EISCONN 3
|
||||||
|
#define EWOULDBLOCK 6
|
||||||
|
#define EALREADY 10
|
||||||
|
#define EAGAIN EWOULDBLOCK
|
||||||
|
#define EINVAL 11
|
||||||
|
#define ENOMEM 18
|
||||||
|
#define EINPROGRESS 22
|
||||||
|
|
||||||
|
#define wiiu_geterrno() (socketlasterr())
|
||||||
|
|
||||||
|
//! C wrapper for our C++ functions
|
||||||
|
int Menu_Main(void);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
283
src/net.c
Normal file
283
src/net.c
Normal file
@ -0,0 +1,283 @@
|
|||||||
|
/*
|
||||||
|
|
||||||
|
Copyright (C) 2008 Joseph Jordan <joe.ftpii@psychlaw.com.au>
|
||||||
|
|
||||||
|
This software is provided 'as-is', without any express or implied warranty.
|
||||||
|
In no event will the authors be held liable for any damages arising from
|
||||||
|
the use of this software.
|
||||||
|
|
||||||
|
Permission is granted to anyone to use this software for any purpose,
|
||||||
|
including commercial applications, and to alter it and redistribute it
|
||||||
|
freely, subject to the following restrictions:
|
||||||
|
|
||||||
|
1.The origin of this software must not be misrepresented; you must not
|
||||||
|
claim that you wrote the original software. If you use this software in a
|
||||||
|
product, an acknowledgment in the product documentation would be
|
||||||
|
appreciated but is not required.
|
||||||
|
|
||||||
|
2.Altered source versions must be plainly marked as such, and must not be
|
||||||
|
misrepresented as being the original software.
|
||||||
|
|
||||||
|
3.This notice may not be removed or altered from any source distribution.
|
||||||
|
|
||||||
|
*/
|
||||||
|
#include <stdint.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <malloc.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <sys/fcntl.h>
|
||||||
|
#include <nsysnet/socket.h>
|
||||||
|
#include "main.h"
|
||||||
|
|
||||||
|
#define MIN(x, y) ((x) < (y) ? (x) : (y))
|
||||||
|
|
||||||
|
#include "net.h"
|
||||||
|
|
||||||
|
#define MAX_NET_BUFFER_SIZE (64*1024)
|
||||||
|
#define MIN_NET_BUFFER_SIZE 4096
|
||||||
|
#define FREAD_BUFFER_SIZE (64*1024)
|
||||||
|
|
||||||
|
extern uint32_t hostIpAddress;
|
||||||
|
|
||||||
|
static uint32_t NET_BUFFER_SIZE = MAX_NET_BUFFER_SIZE;
|
||||||
|
|
||||||
|
#if 0
|
||||||
|
void initialise_network() {
|
||||||
|
printf("Waiting for network to initialise...\n");
|
||||||
|
int32_t result = -1;
|
||||||
|
while (!check_reset_synchronous() && result < 0) {
|
||||||
|
net_deinit();
|
||||||
|
while (!check_reset_synchronous() && (result = net_init()) == -WIIU_EAGAIN);
|
||||||
|
if (result < 0)
|
||||||
|
printf("net_init() failed: [%i] %s, retrying...\n", result, strerror(-result));
|
||||||
|
}
|
||||||
|
if (result >= 0) {
|
||||||
|
uint32_t ip = 0;
|
||||||
|
do {
|
||||||
|
ip = net_gethostip();
|
||||||
|
if (!ip)
|
||||||
|
printf("net_gethostip() failed, retrying...\n");
|
||||||
|
} while (!check_reset_synchronous() && !ip);
|
||||||
|
if (ip) {
|
||||||
|
struct in_addr addr;
|
||||||
|
addr.s_addr = ip;
|
||||||
|
printf("Network initialised. Wii IP address: %s\n", inet_ntoa(addr));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
int32_t network_socket(uint32_t domain,uint32_t type,uint32_t protocol) {
|
||||||
|
int sock = socket(domain, type, protocol);
|
||||||
|
if(sock < 0) {
|
||||||
|
int err = -wiiu_geterrno();
|
||||||
|
return (err < 0) ? err : sock;
|
||||||
|
}
|
||||||
|
return sock;
|
||||||
|
}
|
||||||
|
|
||||||
|
int32_t network_bind(int32_t s,struct sockaddr *name,int32_t namelen) {
|
||||||
|
int res = bind(s, name, namelen);
|
||||||
|
if(res < 0) {
|
||||||
|
int err = -wiiu_geterrno();
|
||||||
|
return (err < 0) ? err : res;
|
||||||
|
}
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
int32_t network_listen(int32_t s,uint32_t backlog) {
|
||||||
|
int res = listen(s, backlog);
|
||||||
|
if(res < 0) {
|
||||||
|
int err = -wiiu_geterrno();
|
||||||
|
return (err < 0) ? err : res;
|
||||||
|
}
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
int32_t network_accept(int32_t s,struct sockaddr *addr,int32_t *addrlen) {
|
||||||
|
int res = accept(s, addr, addrlen);
|
||||||
|
if(res < 0) {
|
||||||
|
int err = -wiiu_geterrno();
|
||||||
|
return (err < 0) ? err : res;
|
||||||
|
}
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
int32_t network_connect(int32_t s,struct sockaddr *addr, int32_t addrlen) {
|
||||||
|
int res = connect(s, addr, addrlen);
|
||||||
|
if(res < 0) {
|
||||||
|
int err = -wiiu_geterrno();
|
||||||
|
return (err < 0) ? err : res;
|
||||||
|
}
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
int32_t network_read(int32_t s,void *mem,int32_t len) {
|
||||||
|
int res = recv(s, mem, len, 0);
|
||||||
|
if(res < 0) {
|
||||||
|
int err = -wiiu_geterrno();
|
||||||
|
return (err < 0) ? err : res;
|
||||||
|
}
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint32_t network_gethostip() {
|
||||||
|
return hostIpAddress;
|
||||||
|
}
|
||||||
|
|
||||||
|
int32_t network_write(int32_t s, const void *mem,int32_t len) {
|
||||||
|
int32_t transfered = 0;
|
||||||
|
|
||||||
|
while(len) {
|
||||||
|
int ret = send(s, mem, len, 0);
|
||||||
|
if(ret < 0) {
|
||||||
|
int err = -wiiu_geterrno();
|
||||||
|
transfered = (err < 0) ? err : ret;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
mem += ret;
|
||||||
|
transfered += ret;
|
||||||
|
len -= ret;
|
||||||
|
}
|
||||||
|
return transfered;
|
||||||
|
}
|
||||||
|
|
||||||
|
int32_t network_close(int32_t s) {
|
||||||
|
if(s < 0)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
return socketclose(s);
|
||||||
|
}
|
||||||
|
|
||||||
|
int32_t set_blocking(int32_t s, bool blocking) {
|
||||||
|
int32_t block = !blocking;
|
||||||
|
setsockopt(s, SOL_SOCKET, SO_NONBLOCK, &block, sizeof(block));
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int32_t network_close_blocking(int32_t s) {
|
||||||
|
set_blocking(s, true);
|
||||||
|
return network_close(s);
|
||||||
|
}
|
||||||
|
|
||||||
|
int32_t create_server(uint16_t port) {
|
||||||
|
int32_t server = network_socket(AF_INET, SOCK_STREAM, IPPROTO_IP);
|
||||||
|
if (server < 0)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
|
||||||
|
set_blocking(server, false);
|
||||||
|
uint32_t enable = 1;
|
||||||
|
setsockopt(server, SOL_SOCKET, SO_REUSEADDR, &enable, sizeof(enable));
|
||||||
|
|
||||||
|
struct sockaddr_in bindAddress;
|
||||||
|
memset(&bindAddress, 0, sizeof(bindAddress));
|
||||||
|
bindAddress.sin_family = AF_INET;
|
||||||
|
bindAddress.sin_port = htons(port);
|
||||||
|
bindAddress.sin_addr.s_addr = htonl(INADDR_ANY);
|
||||||
|
|
||||||
|
int32_t ret;
|
||||||
|
if ((ret = network_bind(server, (struct sockaddr *)&bindAddress, sizeof(bindAddress))) < 0) {
|
||||||
|
network_close(server);
|
||||||
|
//gxprintf("Error binding socket: [%i] %s\n", -ret, strerror(-ret));
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
if ((ret = network_listen(server, 3)) < 0) {
|
||||||
|
network_close(server);
|
||||||
|
//gxprintf("Error listening on socket: [%i] %s\n", -ret, strerror(-ret));
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
return server;
|
||||||
|
}
|
||||||
|
|
||||||
|
typedef int32_t (*transferrer_type)(int32_t s, void *mem, int32_t len);
|
||||||
|
static int32_t transfer_exact(int32_t s, char *buf, int32_t length, transferrer_type transferrer) {
|
||||||
|
int32_t result = 0;
|
||||||
|
int32_t remaining = length;
|
||||||
|
int32_t bytes_transferred;
|
||||||
|
set_blocking(s, true);
|
||||||
|
while (remaining) {
|
||||||
|
try_again_with_smaller_buffer:
|
||||||
|
bytes_transferred = transferrer(s, buf, MIN(remaining, (int) NET_BUFFER_SIZE));
|
||||||
|
if (bytes_transferred > 0) {
|
||||||
|
remaining -= bytes_transferred;
|
||||||
|
buf += bytes_transferred;
|
||||||
|
} else if (bytes_transferred < 0) {
|
||||||
|
if (bytes_transferred == -EINVAL && NET_BUFFER_SIZE == MAX_NET_BUFFER_SIZE) {
|
||||||
|
NET_BUFFER_SIZE = MIN_NET_BUFFER_SIZE;
|
||||||
|
usleep(1000);
|
||||||
|
goto try_again_with_smaller_buffer;
|
||||||
|
}
|
||||||
|
result = bytes_transferred;
|
||||||
|
break;
|
||||||
|
} else {
|
||||||
|
result = -ENODATA;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
set_blocking(s, false);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
int32_t send_exact(int32_t s, char *buf, int32_t length) {
|
||||||
|
return transfer_exact(s, buf, length, (transferrer_type)network_write);
|
||||||
|
}
|
||||||
|
|
||||||
|
int32_t send_from_file(int32_t s, FILE *f) {
|
||||||
|
char * buf = (char *) malloc(FREAD_BUFFER_SIZE);
|
||||||
|
if(!buf)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
int32_t bytes_read;
|
||||||
|
int32_t result = 0;
|
||||||
|
|
||||||
|
bytes_read = fread(buf, 1, FREAD_BUFFER_SIZE, f);
|
||||||
|
if (bytes_read > 0) {
|
||||||
|
result = send_exact(s, buf, bytes_read);
|
||||||
|
if (result < 0)
|
||||||
|
goto end;
|
||||||
|
}
|
||||||
|
if (bytes_read < FREAD_BUFFER_SIZE) {
|
||||||
|
result = -!feof(f);
|
||||||
|
goto end;
|
||||||
|
}
|
||||||
|
free(buf);
|
||||||
|
return -WIIU_EAGAIN;
|
||||||
|
end:
|
||||||
|
free(buf);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
int32_t recv_to_file(int32_t s, FILE *f) {
|
||||||
|
char * buf = (char *) malloc(NET_BUFFER_SIZE);
|
||||||
|
if(!buf)
|
||||||
|
return -1;
|
||||||
|
|
||||||
|
int32_t bytes_read;
|
||||||
|
while (1) {
|
||||||
|
try_again_with_smaller_buffer:
|
||||||
|
bytes_read = network_read(s, buf, NET_BUFFER_SIZE);
|
||||||
|
if (bytes_read < 0) {
|
||||||
|
if (bytes_read == -EINVAL && NET_BUFFER_SIZE == MAX_NET_BUFFER_SIZE) {
|
||||||
|
NET_BUFFER_SIZE = MIN_NET_BUFFER_SIZE;
|
||||||
|
usleep(1000);
|
||||||
|
goto try_again_with_smaller_buffer;
|
||||||
|
}
|
||||||
|
free(buf);
|
||||||
|
return bytes_read;
|
||||||
|
} else if (bytes_read == 0) {
|
||||||
|
free(buf);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int32_t bytes_written = fwrite(buf, 1, bytes_read, f);
|
||||||
|
if (bytes_written < bytes_read) {
|
||||||
|
free(buf);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return -1;
|
||||||
|
}
|
63
src/net.h
Normal file
63
src/net.h
Normal file
@ -0,0 +1,63 @@
|
|||||||
|
/*
|
||||||
|
|
||||||
|
Copyright (C) 2008 Joseph Jordan <joe.ftpii@psychlaw.com.au>
|
||||||
|
|
||||||
|
This software is provided 'as-is', without any express or implied warranty.
|
||||||
|
In no event will the authors be held liable for any damages arising from
|
||||||
|
the use of this software.
|
||||||
|
|
||||||
|
Permission is granted to anyone to use this software for any purpose,
|
||||||
|
including commercial applications, and to alter it and redistribute it
|
||||||
|
freely, subject to the following restrictions:
|
||||||
|
|
||||||
|
1.The origin of this software must not be misrepresented; you must not
|
||||||
|
claim that you wrote the original software. If you use this software in a
|
||||||
|
product, an acknowledgment in the product documentation would be
|
||||||
|
appreciated but is not required.
|
||||||
|
|
||||||
|
2.Altered source versions must be plainly marked as such, and must not be
|
||||||
|
misrepresented as being the original software.
|
||||||
|
|
||||||
|
3.This notice may not be removed or altered from any source distribution.
|
||||||
|
|
||||||
|
*/
|
||||||
|
#ifndef _NET_H_
|
||||||
|
#define _NET_H_
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C"{
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdbool.h>
|
||||||
|
|
||||||
|
#if 0
|
||||||
|
void initialise_network();
|
||||||
|
#endif
|
||||||
|
|
||||||
|
int32_t network_socket(uint32_t domain,uint32_t type,uint32_t protocol);
|
||||||
|
int32_t network_bind(int32_t s,struct sockaddr *name,int32_t namelen);
|
||||||
|
int32_t network_listen(int32_t s,uint32_t backlog);
|
||||||
|
int32_t network_accept(int32_t s,struct sockaddr *addr,int32_t *addrlen);
|
||||||
|
int32_t network_connect(int32_t s,struct sockaddr *,int32_t);
|
||||||
|
int32_t network_read(int32_t s,void *mem,int32_t len);
|
||||||
|
int32_t network_close(int32_t s);
|
||||||
|
uint32_t network_gethostip();
|
||||||
|
|
||||||
|
int32_t set_blocking(int32_t s, bool blocking);
|
||||||
|
|
||||||
|
int32_t network_close_blocking(int32_t s);
|
||||||
|
|
||||||
|
int32_t create_server(uint16_t port);
|
||||||
|
|
||||||
|
int32_t send_exact(int32_t s, char *buf, int32_t length);
|
||||||
|
|
||||||
|
int32_t send_from_file(int32_t s, FILE *f);
|
||||||
|
|
||||||
|
int32_t recv_to_file(int32_t s, FILE *f);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif /* _NET_H_ */
|
30
src/utils/BackgroundThreadWrapper.cpp
Normal file
30
src/utils/BackgroundThreadWrapper.cpp
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
#include "BackgroundThreadWrapper.hpp"
|
||||||
|
#include <malloc.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <coreinit/cache.h>
|
||||||
|
|
||||||
|
#include "logger.h"
|
||||||
|
|
||||||
|
BackgroundThreadWrapper::BackgroundThreadWrapper(int32_t priority): CThread(CThread::eAttributeAffCore2, priority, 0x100000) {
|
||||||
|
}
|
||||||
|
|
||||||
|
BackgroundThreadWrapper::~BackgroundThreadWrapper() {
|
||||||
|
exitThread = 1;
|
||||||
|
DCFlushRange((void*)&exitThread, 4);
|
||||||
|
DEBUG_FUNCTION_LINE("Exit thread\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
void BackgroundThreadWrapper::executeThread() {
|
||||||
|
while (1) {
|
||||||
|
if(exitThread) {
|
||||||
|
DEBUG_FUNCTION_LINE("We want to exit\n");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if(!whileLoop()){
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
DEBUG_FUNCTION_LINE("Exit!\n");
|
||||||
|
}
|
||||||
|
|
31
src/utils/BackgroundThreadWrapper.hpp
Normal file
31
src/utils/BackgroundThreadWrapper.hpp
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "CThread.h"
|
||||||
|
#include "CMutex.h"
|
||||||
|
#include <wut_types.h>
|
||||||
|
|
||||||
|
class BackgroundThreadWrapper: public CThread {
|
||||||
|
public:
|
||||||
|
BackgroundThreadWrapper(int32_t priority);
|
||||||
|
virtual ~BackgroundThreadWrapper();
|
||||||
|
protected:
|
||||||
|
BOOL shouldExit() {
|
||||||
|
return (exitThread == 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
void setThreadPriority(int32_t priority) {
|
||||||
|
this->setThreadPriority(priority);
|
||||||
|
}
|
||||||
|
CMutex mutex;
|
||||||
|
private:
|
||||||
|
void executeThread();
|
||||||
|
|
||||||
|
/**
|
||||||
|
Called when a connection has be accepted.
|
||||||
|
**/
|
||||||
|
virtual BOOL whileLoop() = 0;
|
||||||
|
|
||||||
|
volatile int32_t exitThread = 0;
|
||||||
|
|
||||||
|
|
||||||
|
};
|
69
src/utils/CMutex.h
Normal file
69
src/utils/CMutex.h
Normal file
@ -0,0 +1,69 @@
|
|||||||
|
/****************************************************************************
|
||||||
|
* Copyright (C) 2015 Dimok
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
****************************************************************************/
|
||||||
|
#ifndef _CMUTEX_H_
|
||||||
|
#define _CMUTEX_H_
|
||||||
|
|
||||||
|
#include <malloc.h>
|
||||||
|
#include <coreinit/mutex.h>
|
||||||
|
|
||||||
|
class CMutex
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
CMutex() {
|
||||||
|
pMutex = (OSMutex*) malloc(sizeof(OSMutex));
|
||||||
|
if(!pMutex)
|
||||||
|
return;
|
||||||
|
|
||||||
|
OSInitMutex(pMutex);
|
||||||
|
}
|
||||||
|
virtual ~CMutex() {
|
||||||
|
if(pMutex)
|
||||||
|
free(pMutex);
|
||||||
|
}
|
||||||
|
|
||||||
|
void lock(void) {
|
||||||
|
if(pMutex)
|
||||||
|
OSLockMutex(pMutex);
|
||||||
|
}
|
||||||
|
void unlock(void) {
|
||||||
|
if(pMutex)
|
||||||
|
OSUnlockMutex(pMutex);
|
||||||
|
}
|
||||||
|
BOOL tryLock(void) {
|
||||||
|
if(!pMutex)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
return (OSTryLockMutex(pMutex) != 0);
|
||||||
|
}
|
||||||
|
private:
|
||||||
|
OSMutex *pMutex;
|
||||||
|
};
|
||||||
|
|
||||||
|
class CMutexLock
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
CMutexLock() {
|
||||||
|
mutex.lock();
|
||||||
|
}
|
||||||
|
virtual ~CMutexLock() {
|
||||||
|
mutex.unlock();
|
||||||
|
}
|
||||||
|
private:
|
||||||
|
CMutex mutex;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif // _CMUTEX_H_
|
134
src/utils/CThread.h
Normal file
134
src/utils/CThread.h
Normal file
@ -0,0 +1,134 @@
|
|||||||
|
/****************************************************************************
|
||||||
|
* Copyright (C) 2015 Dimok
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
****************************************************************************/
|
||||||
|
#ifndef CTHREAD_H_
|
||||||
|
#define CTHREAD_H_
|
||||||
|
|
||||||
|
#include <malloc.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <coreinit/systeminfo.h>
|
||||||
|
#include <coreinit/thread.h>
|
||||||
|
#include "utils/logger.h"
|
||||||
|
|
||||||
|
class CThread {
|
||||||
|
public:
|
||||||
|
typedef void (* Callback)(CThread *thread, void *arg);
|
||||||
|
|
||||||
|
//! constructor
|
||||||
|
CThread(int32_t iAttr, int32_t iPriority = 16, int32_t iStackSize = 0x8000, CThread::Callback callback = NULL, void *callbackArg = NULL)
|
||||||
|
: pThread(NULL)
|
||||||
|
, pThreadStack(NULL)
|
||||||
|
, pCallback(callback)
|
||||||
|
, pCallbackArg(callbackArg) {
|
||||||
|
//! save attribute assignment
|
||||||
|
iAttributes = iAttr;
|
||||||
|
//! allocate the thread
|
||||||
|
pThread = (OSThread*)memalign(8, sizeof(OSThread));
|
||||||
|
//! allocate the stack
|
||||||
|
pThreadStack = (uint8_t *) memalign(0x20, iStackSize);
|
||||||
|
//! create the thread
|
||||||
|
if(pThread && pThreadStack)
|
||||||
|
OSCreateThread(pThread, &CThread::threadCallback, 1, (char*)this, pThreadStack+iStackSize, iStackSize, iPriority, iAttributes);
|
||||||
|
}
|
||||||
|
|
||||||
|
//! destructor
|
||||||
|
virtual ~CThread() {
|
||||||
|
shutdownThread();
|
||||||
|
DEBUG_FUNCTION_LINE("END\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
static CThread *create(CThread::Callback callback, void *callbackArg, int32_t iAttr = eAttributeNone, int32_t iPriority = 16, int32_t iStackSize = 0x8000) {
|
||||||
|
return ( new CThread(iAttr, iPriority, iStackSize, callback, callbackArg) );
|
||||||
|
}
|
||||||
|
|
||||||
|
//! Get thread ID
|
||||||
|
virtual void* getThread() const {
|
||||||
|
return pThread;
|
||||||
|
}
|
||||||
|
//! Thread entry function
|
||||||
|
virtual void executeThread(void) {
|
||||||
|
if(pCallback)
|
||||||
|
pCallback(this, pCallbackArg);
|
||||||
|
}
|
||||||
|
//! Suspend thread
|
||||||
|
virtual void suspendThread(void) {
|
||||||
|
if(isThreadSuspended()) return;
|
||||||
|
if(pThread) OSSuspendThread(pThread);
|
||||||
|
}
|
||||||
|
//! Resume thread
|
||||||
|
virtual void resumeThread(void) {
|
||||||
|
if(!isThreadSuspended()) return;
|
||||||
|
if(pThread) OSResumeThread(pThread);
|
||||||
|
}
|
||||||
|
//! Set thread priority
|
||||||
|
virtual void setThreadPriority(int prio) {
|
||||||
|
if(pThread) OSSetThreadPriority(pThread, prio);
|
||||||
|
}
|
||||||
|
//! Check if thread is suspended
|
||||||
|
virtual BOOL isThreadSuspended(void) const {
|
||||||
|
if(pThread) return OSIsThreadSuspended(pThread);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
//! Check if thread is terminated
|
||||||
|
virtual BOOL isThreadTerminated(void) const {
|
||||||
|
if(pThread) return OSIsThreadTerminated(pThread);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
//! Check if thread is running
|
||||||
|
virtual BOOL isThreadRunning(void) const {
|
||||||
|
return !isThreadSuspended() && !isThreadRunning();
|
||||||
|
}
|
||||||
|
//! Shutdown thread
|
||||||
|
virtual void shutdownThread(void) {
|
||||||
|
//! wait for thread to finish
|
||||||
|
if(pThread && !(iAttributes & eAttributeDetach)) {
|
||||||
|
if(isThreadSuspended())
|
||||||
|
resumeThread();
|
||||||
|
|
||||||
|
OSJoinThread(pThread, NULL);
|
||||||
|
}
|
||||||
|
//! free the thread stack buffer
|
||||||
|
if(pThreadStack)
|
||||||
|
free(pThreadStack);
|
||||||
|
if(pThread)
|
||||||
|
free(pThread);
|
||||||
|
|
||||||
|
pThread = NULL;
|
||||||
|
pThreadStack = NULL;
|
||||||
|
}
|
||||||
|
//! Thread attributes
|
||||||
|
enum eCThreadAttributes {
|
||||||
|
eAttributeNone = 0x07,
|
||||||
|
eAttributeAffCore0 = 0x01,
|
||||||
|
eAttributeAffCore1 = 0x02,
|
||||||
|
eAttributeAffCore2 = 0x04,
|
||||||
|
eAttributeDetach = 0x08,
|
||||||
|
eAttributePinnedAff = 0x10
|
||||||
|
};
|
||||||
|
private:
|
||||||
|
static int threadCallback(int argc, const char **argv) {
|
||||||
|
//! After call to start() continue with the internal function
|
||||||
|
((CThread *) argv)->executeThread();
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
int iAttributes;
|
||||||
|
OSThread *pThread;
|
||||||
|
uint8_t *pThreadStack;
|
||||||
|
Callback pCallback;
|
||||||
|
void *pCallbackArg;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
82
src/utils/logger.c
Normal file
82
src/utils/logger.c
Normal file
@ -0,0 +1,82 @@
|
|||||||
|
#include <stdarg.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <errno.h>
|
||||||
|
#include <utils/logger.h>
|
||||||
|
#include <nsysnet/socket.h>
|
||||||
|
#include <coreinit/debug.h>
|
||||||
|
|
||||||
|
#include <coreinit/systeminfo.h>
|
||||||
|
#include <coreinit/thread.h>
|
||||||
|
|
||||||
|
static int log_socket __attribute__((section(".data")))= -1;
|
||||||
|
static struct sockaddr_in connect_addr __attribute__((section(".data")));
|
||||||
|
static volatile int log_lock __attribute__((section(".data"))) = 0;
|
||||||
|
|
||||||
|
void log_init_() {
|
||||||
|
int broadcastEnable = 1;
|
||||||
|
log_socket = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP);
|
||||||
|
if (log_socket < 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
setsockopt(log_socket, SOL_SOCKET, SO_BROADCAST, &broadcastEnable, sizeof(broadcastEnable));
|
||||||
|
|
||||||
|
memset(&connect_addr, 0, sizeof(struct sockaddr_in));
|
||||||
|
connect_addr.sin_family = AF_INET;
|
||||||
|
connect_addr.sin_port = 4405;
|
||||||
|
connect_addr.sin_addr.s_addr = htonl(INADDR_BROADCAST);
|
||||||
|
}
|
||||||
|
|
||||||
|
void log_print_(const char *str) {
|
||||||
|
// socket is always 0 initially as it is in the BSS
|
||||||
|
if(log_socket < 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
while(log_lock)
|
||||||
|
OSSleepTicks(OSMicrosecondsToTicks(1000));
|
||||||
|
log_lock = 1;
|
||||||
|
|
||||||
|
int len = strlen(str);
|
||||||
|
int ret;
|
||||||
|
while (len > 0) {
|
||||||
|
int block = len < 1400 ? len : 1400; // take max 1400 bytes per UDP packet
|
||||||
|
ret = sendto(log_socket, str, block, 0, (struct sockaddr *)&connect_addr, sizeof(struct sockaddr_in));
|
||||||
|
if(ret < 0)
|
||||||
|
break;
|
||||||
|
|
||||||
|
len -= ret;
|
||||||
|
str += ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
log_lock = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void OSFatal_printf(const char *format, ...) {
|
||||||
|
char tmp[512];
|
||||||
|
tmp[0] = 0;
|
||||||
|
va_list va;
|
||||||
|
va_start(va, format);
|
||||||
|
if((vsprintf(tmp, format, va) >= 0)) {
|
||||||
|
OSFatal(tmp);
|
||||||
|
}
|
||||||
|
va_end(va);
|
||||||
|
}
|
||||||
|
|
||||||
|
void log_printf_(const char *format, ...) {
|
||||||
|
if(log_socket < 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
char tmp[512];
|
||||||
|
tmp[0] = 0;
|
||||||
|
|
||||||
|
va_list va;
|
||||||
|
va_start(va, format);
|
||||||
|
if((vsprintf(tmp, format, va) >= 0)) {
|
||||||
|
log_print_(tmp);
|
||||||
|
}
|
||||||
|
va_end(va);
|
||||||
|
}
|
||||||
|
|
38
src/utils/logger.h
Normal file
38
src/utils/logger.h
Normal file
@ -0,0 +1,38 @@
|
|||||||
|
#ifndef __LOGGER_H_
|
||||||
|
#define __LOGGER_H_
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
void log_init_();
|
||||||
|
//void log_deinit_(void);
|
||||||
|
void log_print_(const char *str);
|
||||||
|
void log_printf_(const char *format, ...);
|
||||||
|
void OSFatal_printf(const char *format, ...);
|
||||||
|
|
||||||
|
#define __FILENAME_X__ (strrchr(__FILE__, '\\') ? strrchr(__FILE__, '\\') + 1 : __FILE__)
|
||||||
|
#define __FILENAME__ (strrchr(__FILE__, '/') ? strrchr(__FILE__, '/') + 1 : __FILENAME_X__)
|
||||||
|
|
||||||
|
#define OSFATAL_FUNCTION_LINE(FMT, ARGS...)do { \
|
||||||
|
OSFatal_printf("[%s]%s@L%04d: " FMT "",__FILENAME__,__FUNCTION__, __LINE__, ## ARGS); \
|
||||||
|
} while (0)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
#define log_init() log_init_()
|
||||||
|
//#define log_deinit() log_deinit_()
|
||||||
|
#define log_print(str) log_print_(str)
|
||||||
|
#define log_printf(FMT, ARGS...) log_printf_(FMT, ## ARGS);
|
||||||
|
|
||||||
|
#define DEBUG_FUNCTION_LINE(FMT, ARGS...)do { \
|
||||||
|
log_printf("[%23s]%30s@L%04d: " FMT "",__FILENAME__,__FUNCTION__, __LINE__, ## ARGS); \
|
||||||
|
} while (0)
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
41
src/utils/utils.c
Normal file
41
src/utils/utils.c
Normal file
@ -0,0 +1,41 @@
|
|||||||
|
#include <string.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <stdarg.h>
|
||||||
|
#include <stddef.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <malloc.h>
|
||||||
|
#include <utils/logger.h>
|
||||||
|
|
||||||
|
// https://gist.github.com/ccbrown/9722406
|
||||||
|
void dumpHex(const void* data, size_t size) {
|
||||||
|
char ascii[17];
|
||||||
|
size_t i, j;
|
||||||
|
ascii[16] = '\0';
|
||||||
|
DEBUG_FUNCTION_LINE("0x%08X (0x0000): ", data);
|
||||||
|
for (i = 0; i < size; ++i) {
|
||||||
|
log_printf("%02X ", ((unsigned char*)data)[i]);
|
||||||
|
if (((unsigned char*)data)[i] >= ' ' && ((unsigned char*)data)[i] <= '~') {
|
||||||
|
ascii[i % 16] = ((unsigned char*)data)[i];
|
||||||
|
} else {
|
||||||
|
ascii[i % 16] = '.';
|
||||||
|
}
|
||||||
|
if ((i+1) % 8 == 0 || i+1 == size) {
|
||||||
|
log_printf(" ");
|
||||||
|
if ((i+1) % 16 == 0) {
|
||||||
|
log_printf("| %s \n", ascii);
|
||||||
|
if(i + 1 < size) {
|
||||||
|
DEBUG_FUNCTION_LINE("0x%08X (0x%04X); ", data + i + 1,i+1);
|
||||||
|
}
|
||||||
|
} else if (i+1 == size) {
|
||||||
|
ascii[(i+1) % 16] = '\0';
|
||||||
|
if ((i+1) % 16 <= 8) {
|
||||||
|
log_printf(" ");
|
||||||
|
}
|
||||||
|
for (j = (i+1) % 16; j < 16; ++j) {
|
||||||
|
log_printf(" ");
|
||||||
|
}
|
||||||
|
log_printf("| %s \n", ascii);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
35
src/utils/utils.h
Normal file
35
src/utils/utils.h
Normal file
@ -0,0 +1,35 @@
|
|||||||
|
#ifndef __UTILS_H_
|
||||||
|
#define __UTILS_H_
|
||||||
|
|
||||||
|
#include <malloc.h>
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#define LIMIT(x, min, max) \
|
||||||
|
({ \
|
||||||
|
typeof( x ) _x = x; \
|
||||||
|
typeof( min ) _min = min; \
|
||||||
|
typeof( max ) _max = max; \
|
||||||
|
( ( ( _x ) < ( _min ) ) ? ( _min ) : ( ( _x ) > ( _max ) ) ? ( _max) : ( _x ) ); \
|
||||||
|
})
|
||||||
|
|
||||||
|
#define DegToRad(a) ( (a) * 0.01745329252f )
|
||||||
|
#define RadToDeg(a) ( (a) * 57.29577951f )
|
||||||
|
|
||||||
|
#define ALIGN4(x) (((x) + 3) & ~3)
|
||||||
|
#define ALIGN32(x) (((x) + 31) & ~31)
|
||||||
|
|
||||||
|
#define le16(i) ((((uint16_t) ((i) & 0xFF)) << 8) | ((uint16_t) (((i) & 0xFF00) >> 8)))
|
||||||
|
#define le32(i) ((((uint32_t)le16((i) & 0xFFFF)) << 16) | ((uint32_t)le16(((i) & 0xFFFF0000) >> 16)))
|
||||||
|
#define le64(i) ((((uint64_t)le32((i) & 0xFFFFFFFFLL)) << 32) | ((uint64_t)le32(((i) & 0xFFFFFFFF00000000LL) >> 32)))
|
||||||
|
|
||||||
|
//Needs to have log_init() called beforehand.
|
||||||
|
void dumpHex(const void* data, size_t size);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif // __UTILS_H_
|
125
src/virtualpath.c
Normal file
125
src/virtualpath.c
Normal file
@ -0,0 +1,125 @@
|
|||||||
|
/****************************************************************************
|
||||||
|
* Copyright (C) 2008
|
||||||
|
* Joseph Jordan <joe.ftpii@psychlaw.com.au>
|
||||||
|
*
|
||||||
|
* Copyright (C) 2010
|
||||||
|
* by Dimok
|
||||||
|
*
|
||||||
|
* This software is provided 'as-is', without any express or implied
|
||||||
|
* warranty. In no event will the authors be held liable for any
|
||||||
|
* damages arising from the use of this software.
|
||||||
|
*
|
||||||
|
* Permission is granted to anyone to use this software for any
|
||||||
|
* purpose, including commercial applications, and to alter it and
|
||||||
|
* redistribute it freely, subject to the following restrictions:
|
||||||
|
*
|
||||||
|
* 1. The origin of this software must not be misrepresented; you
|
||||||
|
* must not claim that you wrote the original software. If you use
|
||||||
|
* this software in a product, an acknowledgment in the product
|
||||||
|
* documentation would be appreciated but is not required.
|
||||||
|
*
|
||||||
|
* 2. Altered source versions must be plainly marked as such, and
|
||||||
|
* must not be misrepresented as being the original software.
|
||||||
|
*
|
||||||
|
* 3. This notice may not be removed or altered from any source
|
||||||
|
* distribution.
|
||||||
|
*
|
||||||
|
* for WiiXplorer 2010
|
||||||
|
***************************************************************************/
|
||||||
|
#include <malloc.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include "virtualpath.h"
|
||||||
|
#include "utils/logger.h"
|
||||||
|
|
||||||
|
uint8_t MAX_VIRTUAL_PARTITIONS = 0;
|
||||||
|
VIRTUAL_PARTITION * VIRTUAL_PARTITIONS = NULL;
|
||||||
|
|
||||||
|
void VirtualMountDevice(const char * path)
|
||||||
|
{
|
||||||
|
if(!path)
|
||||||
|
return;
|
||||||
|
|
||||||
|
int i = 0;
|
||||||
|
char name[255];
|
||||||
|
char alias[255];
|
||||||
|
char prefix[255];
|
||||||
|
bool namestop = false;
|
||||||
|
|
||||||
|
alias[0] = '/';
|
||||||
|
|
||||||
|
do
|
||||||
|
{
|
||||||
|
if(path[i] == ':')
|
||||||
|
namestop = true;
|
||||||
|
|
||||||
|
if(!namestop)
|
||||||
|
{
|
||||||
|
name[i] = path[i];
|
||||||
|
name[i+1] = '\0';
|
||||||
|
alias[i+1] = path[i];
|
||||||
|
alias[i+2] = '\0';
|
||||||
|
}
|
||||||
|
|
||||||
|
prefix[i] = path[i];
|
||||||
|
prefix[i+1] = '\0';
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
while(path[i-1] != '/');
|
||||||
|
AddVirtualPath(name, alias, prefix);
|
||||||
|
}
|
||||||
|
|
||||||
|
void AddVirtualPath(const char *name, const char *alias, const char *prefix)
|
||||||
|
{
|
||||||
|
if(!VIRTUAL_PARTITIONS)
|
||||||
|
VIRTUAL_PARTITIONS = (VIRTUAL_PARTITION *) malloc(sizeof(VIRTUAL_PARTITION));
|
||||||
|
|
||||||
|
VIRTUAL_PARTITION * tmp = realloc(VIRTUAL_PARTITIONS, sizeof(VIRTUAL_PARTITION)*(MAX_VIRTUAL_PARTITIONS+1));
|
||||||
|
if(!tmp)
|
||||||
|
{
|
||||||
|
free(VIRTUAL_PARTITIONS);
|
||||||
|
MAX_VIRTUAL_PARTITIONS = 0;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
VIRTUAL_PARTITIONS = tmp;
|
||||||
|
|
||||||
|
VIRTUAL_PARTITIONS[MAX_VIRTUAL_PARTITIONS].name = strdup(name);
|
||||||
|
VIRTUAL_PARTITIONS[MAX_VIRTUAL_PARTITIONS].alias = strdup(alias);
|
||||||
|
VIRTUAL_PARTITIONS[MAX_VIRTUAL_PARTITIONS].prefix = strdup(prefix);
|
||||||
|
VIRTUAL_PARTITIONS[MAX_VIRTUAL_PARTITIONS].inserted = true;
|
||||||
|
|
||||||
|
MAX_VIRTUAL_PARTITIONS++;
|
||||||
|
}
|
||||||
|
|
||||||
|
void MountVirtualDevices()
|
||||||
|
{
|
||||||
|
VirtualMountDevice("fs:/");
|
||||||
|
VirtualMountDevice("slccmpt01:/");
|
||||||
|
VirtualMountDevice("storage_odd_tickets:/");
|
||||||
|
VirtualMountDevice("storage_odd_updates:/");
|
||||||
|
VirtualMountDevice("storage_odd_content:/");
|
||||||
|
VirtualMountDevice("storage_odd_content2:/");
|
||||||
|
VirtualMountDevice("storage_slc:/");
|
||||||
|
VirtualMountDevice("storage_mlc:/");
|
||||||
|
VirtualMountDevice("storage_usb:/");
|
||||||
|
VirtualMountDevice("usb:/");
|
||||||
|
}
|
||||||
|
|
||||||
|
void UnmountVirtualPaths()
|
||||||
|
{
|
||||||
|
uint32_t i = 0;
|
||||||
|
for(i = 0; i < MAX_VIRTUAL_PARTITIONS; i++)
|
||||||
|
{
|
||||||
|
if(VIRTUAL_PARTITIONS[i].name)
|
||||||
|
free(VIRTUAL_PARTITIONS[i].name);
|
||||||
|
if(VIRTUAL_PARTITIONS[i].alias)
|
||||||
|
free(VIRTUAL_PARTITIONS[i].alias);
|
||||||
|
if(VIRTUAL_PARTITIONS[i].prefix)
|
||||||
|
free(VIRTUAL_PARTITIONS[i].prefix);
|
||||||
|
}
|
||||||
|
|
||||||
|
if(VIRTUAL_PARTITIONS)
|
||||||
|
free(VIRTUAL_PARTITIONS);
|
||||||
|
VIRTUAL_PARTITIONS = NULL;
|
||||||
|
MAX_VIRTUAL_PARTITIONS = 0;
|
||||||
|
}
|
59
src/virtualpath.h
Normal file
59
src/virtualpath.h
Normal file
@ -0,0 +1,59 @@
|
|||||||
|
/****************************************************************************
|
||||||
|
* Copyright (C) 2010
|
||||||
|
* by Dimok
|
||||||
|
*
|
||||||
|
* Original VIRTUAL_PART Struct
|
||||||
|
* Copyright (C) 2008
|
||||||
|
* Joseph Jordan <joe.ftpii@psychlaw.com.au>
|
||||||
|
*
|
||||||
|
* This software is provided 'as-is', without any express or implied
|
||||||
|
* warranty. In no event will the authors be held liable for any
|
||||||
|
* damages arising from the use of this software.
|
||||||
|
*
|
||||||
|
* Permission is granted to anyone to use this software for any
|
||||||
|
* purpose, including commercial applications, and to alter it and
|
||||||
|
* redistribute it freely, subject to the following restrictions:
|
||||||
|
*
|
||||||
|
* 1. The origin of this software must not be misrepresented; you
|
||||||
|
* must not claim that you wrote the original software. If you use
|
||||||
|
* this software in a product, an acknowledgment in the product
|
||||||
|
* documentation would be appreciated but is not required.
|
||||||
|
*
|
||||||
|
* 2. Altered source versions must be plainly marked as such, and
|
||||||
|
* must not be misrepresented as being the original software.
|
||||||
|
*
|
||||||
|
* 3. This notice may not be removed or altered from any source
|
||||||
|
* distribution.
|
||||||
|
*
|
||||||
|
* for WiiXplorer 2010
|
||||||
|
***************************************************************************/
|
||||||
|
#ifndef _VIRTUALPATH_H_
|
||||||
|
#define _VIRTUALPATH_H_
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C"{
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include <stdint.h>
|
||||||
|
#include <stdbool.h>
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
char *name;
|
||||||
|
char *alias;
|
||||||
|
char *prefix;
|
||||||
|
bool inserted;
|
||||||
|
} VIRTUAL_PARTITION;
|
||||||
|
|
||||||
|
extern VIRTUAL_PARTITION * VIRTUAL_PARTITIONS;
|
||||||
|
extern uint8_t MAX_VIRTUAL_PARTITIONS;
|
||||||
|
|
||||||
|
void VirtualMountDevice(const char * devicepath);
|
||||||
|
void AddVirtualPath(const char *name, const char *alias, const char *prefix);
|
||||||
|
void MountVirtualDevices();
|
||||||
|
void UnmountVirtualPaths();
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif /* _VIRTUALPART_H_ */
|
346
src/vrt.c
Normal file
346
src/vrt.c
Normal file
@ -0,0 +1,346 @@
|
|||||||
|
/*
|
||||||
|
|
||||||
|
Copyright (C) 2008 Joseph Jordan <joe.ftpii@psychlaw.com.au>
|
||||||
|
This work is derived from Daniel Ehlers' <danielehlers@mindeye.net> srg_vrt branch.
|
||||||
|
|
||||||
|
This software is provided 'as-is', without any express or implied warranty.
|
||||||
|
In no event will the authors be held liable for any damages arising from
|
||||||
|
the use of this software.
|
||||||
|
|
||||||
|
Permission is granted to anyone to use this software for any purpose,
|
||||||
|
including commercial applications, and to alter it and redistribute it
|
||||||
|
freely, subject to the following restrictions:
|
||||||
|
|
||||||
|
1.The origin of this software must not be misrepresented; you must not
|
||||||
|
claim that you wrote the original software. If you use this software in a
|
||||||
|
product, an acknowledgment in the product documentation would be
|
||||||
|
appreciated but is not required.
|
||||||
|
|
||||||
|
2.Altered source versions must be plainly marked as such, and must not be
|
||||||
|
misrepresented as being the original software.
|
||||||
|
|
||||||
|
3.This notice may not be removed or altered from any source distribution.
|
||||||
|
|
||||||
|
*/
|
||||||
|
#include <errno.h>
|
||||||
|
#include <malloc.h>
|
||||||
|
#include <stdarg.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <sys/dirent.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include "main.h"
|
||||||
|
|
||||||
|
#include "virtualpath.h"
|
||||||
|
#include "vrt.h"
|
||||||
|
|
||||||
|
static char *virtual_abspath(char *virtual_cwd, char *virtual_path) {
|
||||||
|
char *path;
|
||||||
|
if (virtual_path[0] == '/') {
|
||||||
|
path = virtual_path;
|
||||||
|
} else {
|
||||||
|
size_t path_size = strlen(virtual_cwd) + strlen(virtual_path) + 1;
|
||||||
|
if (path_size > MAXPATHLEN || !(path = malloc(path_size))) return NULL;
|
||||||
|
strcpy(path, virtual_cwd);
|
||||||
|
strcat(path, virtual_path);
|
||||||
|
}
|
||||||
|
|
||||||
|
char *normalised_path = malloc(strlen(path) + 1);
|
||||||
|
if (!normalised_path) goto end;
|
||||||
|
*normalised_path = '\0';
|
||||||
|
char *curr_dir = normalised_path;
|
||||||
|
|
||||||
|
uint32_t state = 0; // 0:start, 1:slash, 2:dot, 3:dotdot
|
||||||
|
char *token = path;
|
||||||
|
while (1) {
|
||||||
|
switch (state) {
|
||||||
|
case 0:
|
||||||
|
if (*token == '/') {
|
||||||
|
state = 1;
|
||||||
|
curr_dir = normalised_path + strlen(normalised_path);
|
||||||
|
strncat(normalised_path, token, 1);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 1:
|
||||||
|
if (*token == '.') state = 2;
|
||||||
|
else if (*token != '/') state = 0;
|
||||||
|
break;
|
||||||
|
case 2:
|
||||||
|
if (*token == '/' || !*token) {
|
||||||
|
state = 1;
|
||||||
|
*(curr_dir + 1) = '\0';
|
||||||
|
} else if (*token == '.') state = 3;
|
||||||
|
else state = 0;
|
||||||
|
break;
|
||||||
|
case 3:
|
||||||
|
if (*token == '/' || !*token) {
|
||||||
|
state = 1;
|
||||||
|
*curr_dir = '\0';
|
||||||
|
char *prev_dir = strrchr(normalised_path, '/');
|
||||||
|
if (prev_dir) curr_dir = prev_dir;
|
||||||
|
else *curr_dir = '/';
|
||||||
|
*(curr_dir + 1) = '\0';
|
||||||
|
} else state = 0;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (!*token) break;
|
||||||
|
if (state == 0 || *token != '/') strncat(normalised_path, token, 1);
|
||||||
|
token++;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint32_t end = strlen(normalised_path);
|
||||||
|
while (end > 1 && normalised_path[end - 1] == '/') {
|
||||||
|
normalised_path[--end] = '\x00';
|
||||||
|
}
|
||||||
|
|
||||||
|
end:
|
||||||
|
if (path != virtual_path) free(path);
|
||||||
|
return normalised_path;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
Converts a client-visible path to a real absolute path
|
||||||
|
E.g. "/sd/foo" -> "sd:/foo"
|
||||||
|
"/sd" -> "sd:/"
|
||||||
|
"/sd/../usb" -> "usb:/"
|
||||||
|
The resulting path will fit in an array of size MAXPATHLEN
|
||||||
|
Returns NULL to indicate that the client-visible path is invalid
|
||||||
|
*/
|
||||||
|
char *to_real_path(char *virtual_cwd, char *virtual_path) {
|
||||||
|
errno = ENOENT;
|
||||||
|
if (strchr(virtual_path, ':')) {
|
||||||
|
return NULL; // colon is not allowed in virtual path, i've decided =P
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual_path = virtual_abspath(virtual_cwd, virtual_path);
|
||||||
|
if (!virtual_path) return NULL;
|
||||||
|
|
||||||
|
char *path = NULL;
|
||||||
|
char *rest = virtual_path;
|
||||||
|
|
||||||
|
if (!strcmp("/", virtual_path)) {
|
||||||
|
// indicate vfs-root with ""
|
||||||
|
path = "";
|
||||||
|
goto end;
|
||||||
|
}
|
||||||
|
|
||||||
|
const char *prefix = NULL;
|
||||||
|
uint32_t i;
|
||||||
|
for (i = 0; i < MAX_VIRTUAL_PARTITIONS; i++) {
|
||||||
|
VIRTUAL_PARTITION *partition = VIRTUAL_PARTITIONS + i;
|
||||||
|
const char *alias = partition->alias;
|
||||||
|
size_t alias_len = strlen(alias);
|
||||||
|
if (!strcasecmp(alias, virtual_path) || (!strncasecmp(alias, virtual_path, alias_len) && virtual_path[alias_len] == '/')) {
|
||||||
|
prefix = partition->prefix;
|
||||||
|
rest += alias_len;
|
||||||
|
if (*rest == '/') rest++;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!prefix) {
|
||||||
|
errno = ENODEV;
|
||||||
|
goto end;
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t real_path_size = strlen(prefix) + strlen(rest) + 1;
|
||||||
|
if (real_path_size > MAXPATHLEN) goto end;
|
||||||
|
|
||||||
|
path = malloc(real_path_size);
|
||||||
|
if (!path) goto end;
|
||||||
|
strcpy(path, prefix);
|
||||||
|
strcat(path, rest);
|
||||||
|
|
||||||
|
end:
|
||||||
|
free(virtual_path);
|
||||||
|
return path;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int checkdir(char *path) {
|
||||||
|
DIR *dir = opendir(path);
|
||||||
|
if(dir)
|
||||||
|
{
|
||||||
|
closedir(dir);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
typedef void * (*path_func)(char *path, ...);
|
||||||
|
|
||||||
|
static void *with_virtual_path(void *virtual_cwd, void *void_f, char *virtual_path, int32_t failed, ...) {
|
||||||
|
char *path = to_real_path(virtual_cwd, virtual_path);
|
||||||
|
if (!path || !*path) return (void *)failed;
|
||||||
|
|
||||||
|
path_func f = (path_func)void_f;
|
||||||
|
va_list ap;
|
||||||
|
void *args[3];
|
||||||
|
unsigned int num_args = 0;
|
||||||
|
va_start(ap, failed);
|
||||||
|
do {
|
||||||
|
void *arg = va_arg(ap, void *);
|
||||||
|
if (!arg) break;
|
||||||
|
args[num_args++] = arg;
|
||||||
|
} while (1);
|
||||||
|
va_end(ap);
|
||||||
|
|
||||||
|
void *result;
|
||||||
|
switch (num_args) {
|
||||||
|
case 0: result = f(path); break;
|
||||||
|
case 1: result = f(path, args[0]); break;
|
||||||
|
case 2: result = f(path, args[0], args[1]); break;
|
||||||
|
case 3: result = f(path, args[0], args[1], args[2]); break;
|
||||||
|
default: result = (void *)failed; break;
|
||||||
|
}
|
||||||
|
|
||||||
|
free(path);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
FILE *vrt_fopen(char *cwd, char *path, char *mode) {
|
||||||
|
return with_virtual_path(cwd, fopen, path, 0, mode, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
int vrt_stat(char *cwd, char *path, struct stat *st) {
|
||||||
|
char *real_path = to_real_path(cwd, path);
|
||||||
|
if (!real_path)
|
||||||
|
{
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
else if (!*real_path || (strcmp(path, ".") == 0) || (strlen(cwd) == 1) || ((strlen(cwd) > 1) && (strcmp(path, "..") == 0)))
|
||||||
|
{
|
||||||
|
st->st_mode = S_IFDIR;
|
||||||
|
st->st_size = 31337;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
free(real_path);
|
||||||
|
return (int)with_virtual_path(cwd, stat, path, -1, st, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int vrt_checkdir(char *cwd, char *path) {
|
||||||
|
char *real_path = to_real_path(cwd, path);
|
||||||
|
if (!real_path)
|
||||||
|
{
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
else if (!*real_path || (strcmp(path, ".") == 0) || (strlen(cwd) == 1) || ((strlen(cwd) > 1) && (strcmp(path, "..") == 0)))
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
free(real_path);
|
||||||
|
return (int)with_virtual_path(cwd, checkdir, path, -1, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
int vrt_chdir(char *cwd, char *path) {
|
||||||
|
|
||||||
|
if (vrt_checkdir(cwd, path)) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
char *abspath = virtual_abspath(cwd, path);
|
||||||
|
if (!abspath) {
|
||||||
|
errno = ENOMEM;
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
strcpy(cwd, abspath);
|
||||||
|
if (cwd[1]) strcat(cwd, "/");
|
||||||
|
free(abspath);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int vrt_unlink(char *cwd, char *path) {
|
||||||
|
return (int)with_virtual_path(cwd, unlink, path, -1, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
int vrt_mkdir(char *cwd, char *path, mode_t mode) {
|
||||||
|
return (int)with_virtual_path(cwd, mkdir, path, -1, mode, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
int vrt_rename(char *cwd, char *from_path, char *to_path) {
|
||||||
|
char *real_to_path = to_real_path(cwd, to_path);
|
||||||
|
if (!real_to_path || !*real_to_path) return -1;
|
||||||
|
int result = (int)with_virtual_path(cwd, rename, from_path, -1, real_to_path, NULL);
|
||||||
|
free(real_to_path);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
When in vfs-root this creates a fake DIR_ITER.
|
||||||
|
*/
|
||||||
|
DIR_P *vrt_opendir(char *cwd, char *path)
|
||||||
|
{
|
||||||
|
char *real_path = to_real_path(cwd, path);
|
||||||
|
if (!real_path) return NULL;
|
||||||
|
|
||||||
|
DIR_P *iter = malloc(sizeof(DIR_P));
|
||||||
|
if (!iter)
|
||||||
|
{
|
||||||
|
if (*real_path != 0)
|
||||||
|
free(real_path);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
iter->virt_root = 0;
|
||||||
|
iter->path = real_path;
|
||||||
|
|
||||||
|
if (*iter->path == 0) {
|
||||||
|
iter->dir = malloc(sizeof(DIR));
|
||||||
|
if(!iter->dir) {
|
||||||
|
// root path is not allocated
|
||||||
|
free(iter);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
memset(iter->dir, 0, sizeof(DIR));
|
||||||
|
iter->virt_root = 1; // we are at the virtual root
|
||||||
|
return iter;
|
||||||
|
}
|
||||||
|
|
||||||
|
iter->dir = with_virtual_path(cwd, opendir, path, 0, NULL);
|
||||||
|
if(!iter->dir)
|
||||||
|
{
|
||||||
|
free(iter->path);
|
||||||
|
free(iter);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
return iter;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
Yields virtual aliases when pDir->virt_root
|
||||||
|
*/
|
||||||
|
struct dirent *vrt_readdir(DIR_P *pDir) {
|
||||||
|
if(!pDir || !pDir->dir) return NULL;
|
||||||
|
|
||||||
|
DIR *iter = pDir->dir;
|
||||||
|
if (pDir->virt_root) {
|
||||||
|
for (; (uint32_t)iter->position < MAX_VIRTUAL_PARTITIONS; iter->position++) {
|
||||||
|
VIRTUAL_PARTITION *partition = VIRTUAL_PARTITIONS + (int)iter->position;
|
||||||
|
if (partition->inserted) {
|
||||||
|
iter->fileData.d_type = DT_DIR;
|
||||||
|
strcpy(iter->fileData.d_name, partition->alias + 1);
|
||||||
|
iter->position++;
|
||||||
|
return &iter->fileData;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
return readdir(iter);
|
||||||
|
}
|
||||||
|
|
||||||
|
int vrt_closedir(DIR_P *iter) {
|
||||||
|
if(!iter) return -1;
|
||||||
|
|
||||||
|
if(iter->dir)
|
||||||
|
{
|
||||||
|
if (iter->virt_root)
|
||||||
|
free(iter->dir);
|
||||||
|
else
|
||||||
|
closedir(iter->dir);
|
||||||
|
}
|
||||||
|
|
||||||
|
// root path is not allocated
|
||||||
|
if(iter->path && *iter->path != 0)
|
||||||
|
free(iter->path);
|
||||||
|
|
||||||
|
free(iter);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
58
src/vrt.h
Normal file
58
src/vrt.h
Normal file
@ -0,0 +1,58 @@
|
|||||||
|
/*
|
||||||
|
|
||||||
|
Copyright (C) 2008 Joseph Jordan <joe.ftpii@psychlaw.com.au>
|
||||||
|
This work is derived from Daniel Ehlers' <danielehlers@mindeye.net> srg_vrt branch.
|
||||||
|
|
||||||
|
This software is provided 'as-is', without any express or implied warranty.
|
||||||
|
In no event will the authors be held liable for any damages arising from
|
||||||
|
the use of this software.
|
||||||
|
|
||||||
|
Permission is granted to anyone to use this software for any purpose,
|
||||||
|
including commercial applications, and to alter it and redistribute it
|
||||||
|
freely, subject to the following restrictions:
|
||||||
|
|
||||||
|
1.The origin of this software must not be misrepresented; you must not
|
||||||
|
claim that you wrote the original software. If you use this software in a
|
||||||
|
product, an acknowledgment in the product documentation would be
|
||||||
|
appreciated but is not required.
|
||||||
|
|
||||||
|
2.Altered source versions must be plainly marked as such, and must not be
|
||||||
|
misrepresented as being the original software.
|
||||||
|
|
||||||
|
3.This notice may not be removed or altered from any source distribution.
|
||||||
|
|
||||||
|
*/
|
||||||
|
#ifndef _VRT_H_
|
||||||
|
#define _VRT_H_
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C"{
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <sys/dirent.h>
|
||||||
|
|
||||||
|
typedef struct
|
||||||
|
{
|
||||||
|
DIR *dir;
|
||||||
|
char *path;
|
||||||
|
uint8_t virt_root;
|
||||||
|
} DIR_P;
|
||||||
|
|
||||||
|
char *to_real_path(char *virtual_cwd, char *virtual_path);
|
||||||
|
|
||||||
|
FILE *vrt_fopen(char *cwd, char *path, char *mode);
|
||||||
|
int vrt_stat(char *cwd, char *path, struct stat *st);
|
||||||
|
int vrt_chdir(char *cwd, char *path);
|
||||||
|
int vrt_unlink(char *cwd, char *path);
|
||||||
|
int vrt_mkdir(char *cwd, char *path, mode_t mode);
|
||||||
|
int vrt_rename(char *cwd, char *from_path, char *to_path);
|
||||||
|
DIR_P *vrt_opendir(char *cwd, char *path);
|
||||||
|
struct dirent *vrt_readdir(DIR_P *iter);
|
||||||
|
int vrt_closedir(DIR_P *iter);
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif /* _VRT_H_ */
|
Loading…
x
Reference in New Issue
Block a user