Update for template changes.

This commit is contained in:
Steven Smith 2015-01-21 21:56:25 -08:00
parent 797143746e
commit 61cc6f21f6
24 changed files with 513 additions and 407 deletions

10
.gitignore vendored
View File

@ -1,9 +1,5 @@
.idea .idea
build
CMakeLists.txt CMakeLists.txt
*.3dsx
*.elf build
*.smdh output
*.zip
*.cia
*.3ds

View File

@ -9,13 +9,11 @@ endif
TOPDIR ?= $(CURDIR) TOPDIR ?= $(CURDIR)
include $(DEVKITARM)/3ds_rules include $(DEVKITARM)/3ds_rules
APP_ID = FBI
APP_TITLE = FBI APP_TITLE = FBI
APP_DESCRIPTION = Open source CIA installer. APP_DESCRIPTION = Open source CIA installer.
APP_AUTHOR = Steveice10 APP_AUTHOR = Steveice10
#--------------------------------------------------------------------------------- #---------------------------------------------------------------------------------
# TARGET is the name of the output
# BUILD is the directory where object files & intermediate files will be placed # BUILD is the directory where object files & intermediate files will be placed
# SOURCES is a list of directories containing source code # SOURCES is a list of directories containing source code
# DATA is a list of directories containing data files # DATA is a list of directories containing data files
@ -31,7 +29,6 @@ APP_AUTHOR = Steveice10
# - icon.png # - icon.png
# - <libctru folder>/default_icon.png # - <libctru folder>/default_icon.png
#--------------------------------------------------------------------------------- #---------------------------------------------------------------------------------
TARGET := $(APP_ID)
BUILD := build BUILD := build
SOURCES := source SOURCES := source
DATA := data DATA := data
@ -65,12 +62,9 @@ LIBS := -lctru -lm
#--------------------------------------------------------------------------------- #---------------------------------------------------------------------------------
LIBDIRS := $(CTRULIB) ./lib LIBDIRS := $(CTRULIB) ./lib
MAKEROM = $(TOPDIR)/resources/tools/makerom MAKEROM = $(TOPDIR)/tools/makerom
BANNER_TOOL = $(TOPDIR)/resources/tools/banner BANNER_TOOL = $(TOPDIR)/tools/banner
PREPARE_BANNER = python2 banner.py CREATE_BANNER = python create.py
PREPARE_ICON24 = python2 icon.py
PREPARE_ICON48 = python2 icon.py
CREATE_BANNER = python2 create.py
#--------------------------------------------------------------------------------- #---------------------------------------------------------------------------------
@ -80,7 +74,10 @@ CREATE_BANNER = python2 create.py
ifneq ($(BUILD),$(notdir $(CURDIR))) ifneq ($(BUILD),$(notdir $(CURDIR)))
#--------------------------------------------------------------------------------- #---------------------------------------------------------------------------------
export OUTPUT := $(CURDIR)/$(TARGET) null :=
SPACE := $(null) $(null)
export OUTPUT_D := $(CURDIR)/output
export OUTPUT := $(OUTPUT_D)/$(subst $(SPACE),,$(APP_TITLE))
export TOPDIR := $(CURDIR) export TOPDIR := $(CURDIR)
export VPATH := $(foreach dir,$(SOURCES),$(CURDIR)/$(dir)) \ export VPATH := $(foreach dir,$(SOURCES),$(CURDIR)/$(dir)) \
@ -116,18 +113,7 @@ export INCLUDE := $(foreach dir,$(INCLUDES),-I$(CURDIR)/$(dir)) \
export LIBPATHS := $(foreach dir,$(LIBDIRS),-L$(dir)/lib) export LIBPATHS := $(foreach dir,$(LIBDIRS),-L$(dir)/lib)
ifeq ($(strip $(ICON)),) export APP_ICON := $(TOPDIR)/$(ICON)
icons := $(wildcard *.png)
ifneq (,$(findstring $(TARGET).png,$(icons)))
export APP_ICON := $(TOPDIR)/$(TARGET).png
else
ifneq (,$(findstring icon.png,$(icons)))
export APP_ICON := $(TOPDIR)/icon.png
endif
endif
else
export APP_ICON := $(TOPDIR)/$(ICON)
endif
.PHONY: $(BUILD) clean all .PHONY: $(BUILD) clean all
@ -141,7 +127,7 @@ $(BUILD):
#--------------------------------------------------------------------------------- #---------------------------------------------------------------------------------
clean: clean:
@echo clean ... @echo clean ...
@rm -fr $(BUILD) $(TARGET).3dsx $(OUTPUT).smdh $(TARGET).elf $(TARGET).3ds $(TARGET).cia $(BANNER_TOOL)/banner.bnr $(BANNER_TOOL)/icon.icn @rm -fr $(BUILD) $(OUTPUT_D)
#--------------------------------------------------------------------------------- #---------------------------------------------------------------------------------
@ -154,40 +140,30 @@ DEPENDS := $(OFILES:.o=.d)
#--------------------------------------------------------------------------------- #---------------------------------------------------------------------------------
ifeq ($(strip $(NO_SMDH)),) ifeq ($(strip $(NO_SMDH)),)
.PHONY: all .PHONY: all
all : $(OUTPUT).3dsx $(OUTPUT).smdh $(OUTPUT).cia $(OUTPUT).3ds all : $(OUTPUT_D) $(OUTPUT).3dsx $(OUTPUT).smdh $(OUTPUT).cia $(OUTPUT).3ds
endif endif
$(OUTPUT_D) :
@[ -d $@ ] || mkdir -p $@
$(OUTPUT).3dsx : $(OUTPUT).elf $(OUTPUT).3dsx : $(OUTPUT).elf
$(OUTPUT).elf : $(OFILES) $(OUTPUT).elf : $(OFILES)
$(BANNER_TOOL)/banner.bnr: $(TOPDIR)/resources/banner.png $(TOPDIR)/resources/icon24.png $(TOPDIR)/resources/icon48.png banner.bnr: $(TOPDIR)/resources/icon24.png $(TOPDIR)/resources/icon48.png $(TOPDIR)/resources/banner.png
cp $(TOPDIR)/resources/banner.png $(BANNER_TOOL)/banner/banner.png cd $(BANNER_TOOL); $(CREATE_BANNER) "$(APP_TITLE)" "$(APP_TITLE)" "$(APP_AUTHOR)" $(TOPDIR)/resources/icon24.png $(TOPDIR)/resources/icon48.png $(TOPDIR)/resources/banner.png $(TOPDIR)/$(BUILD)/icon.icn $(TOPDIR)/$(BUILD)/banner.bnr
cp $(TOPDIR)/resources/icon24.png $(BANNER_TOOL)/icon24/icon24.png @echo "built ... banner files"
cp $(TOPDIR)/resources/icon48.png $(BANNER_TOOL)/icon48/icon48.png
cd $(BANNER_TOOL)/banner; $(PREPARE_BANNER)
cd $(BANNER_TOOL)/icon24; $(PREPARE_ICON24)
cd $(BANNER_TOOL)/icon48; $(PREPARE_ICON48)
cd $(BANNER_TOOL); $(CREATE_BANNER)
rm $(BANNER_TOOL)/banner/banner.png
rm $(BANNER_TOOL)/banner/banner.cgfx
rm $(BANNER_TOOL)/banner/compressed.cgfx
rm $(BANNER_TOOL)/banner/banner.cbmd
rm $(BANNER_TOOL)/icon24/icon24.png
rm $(BANNER_TOOL)/icon24/icon24.ctpk
rm $(BANNER_TOOL)/icon48/icon48.png
rm $(BANNER_TOOL)/icon48/icon48.ctpk
$(BANNER_TOOL)/icon.icn: $(BANNER_TOOL)/banner.bnr icon.icn: banner.bnr
$(OUTPUT).cia: $(OUTPUT).elf $(TOPDIR)/resources/cia.rsf $(BANNER_TOOL)/banner.bnr $(BANNER_TOOL)/icon.icn stripped.elf: $(OUTPUT).elf
@cp $(OUTPUT).elf $(TARGET)_stripped.elf @cp $(OUTPUT).elf stripped.elf
@$(PREFIX)strip $(TARGET)_stripped.elf @$(PREFIX)strip stripped.elf
$(MAKEROM) -f cia -o $(OUTPUT).cia -rsf $(TOPDIR)/resources/cia.rsf -target t -exefslogo -elf $(TARGET)_stripped.elf -icon $(BANNER_TOOL)/icon.icn -banner $(BANNER_TOOL)/banner.bnr
$(OUTPUT).cia: stripped.elf $(TOPDIR)/resources/cia.rsf banner.bnr icon.icn
$(MAKEROM) -f cia -o $(OUTPUT).cia -rsf $(TOPDIR)/resources/cia.rsf -target t -exefslogo -elf stripped.elf -icon icon.icn -banner banner.bnr
@echo "built ... $(notdir $@)" @echo "built ... $(notdir $@)"
$(OUTPUT).3ds: $(OUTPUT).elf $(TOPDIR)/resources/3ds.rsf $(BANNER_TOOL)/banner.bnr $(BANNER_TOOL)/icon.icn $(OUTPUT).3ds: stripped.elf $(TOPDIR)/resources/3ds.rsf banner.bnr icon.icn
@cp $(OUTPUT).elf $(TARGET)_stripped.elf $(MAKEROM) -f cci -o $(OUTPUT).3ds -rsf $(TOPDIR)/resources/3ds.rsf -target d -exefslogo -elf stripped.elf -icon icon.icn -banner banner.bnr
@$(PREFIX)strip $(TARGET)_stripped.elf
$(MAKEROM) -f cci -o $(OUTPUT).3ds -rsf $(TOPDIR)/resources/3ds.rsf -target d -exefslogo -elf $(TARGET)_stripped.elf -icon $(BANNER_TOOL)/icon.icn -banner $(BANNER_TOOL)/banner.bnr
@echo "built ... $(notdir $@)" @echo "built ... $(notdir $@)"
#--------------------------------------------------------------------------------- #---------------------------------------------------------------------------------

View File

@ -15,7 +15,7 @@ TitleInfo:
Category : Application Category : Application
CardInfo: CardInfo:
MediaSize : 128MB # 128MB / 256MB / 512MB / 1GB / 2GB / 4GB MediaSize : 128MB # 128MB / 256MB / 512MB / 1GB / 2GB / 4GB / 8GB / 16GB / 32GB
MediaType : Card1 # Card1 / Card2 MediaType : Card1 # Card1 / Card2
CardDevice : None # NorFlash(Pick this if you use savedata) / None CardDevice : None # NorFlash(Pick this if you use savedata) / None
@ -26,6 +26,7 @@ Option:
EnableCrypt : true # Enables encryption for NCCH and CIA EnableCrypt : true # Enables encryption for NCCH and CIA
EnableCompress : true # Compresses exefs code EnableCompress : true # Compresses exefs code
ExeFs: # these are the program segments from the ELF, check your elf for the appropriate segment names ExeFs: # these are the program segments from the ELF, check your elf for the appropriate segment names
ReadOnly: ReadOnly:
- .rodata - .rodata
@ -42,10 +43,10 @@ PlainRegion: # only used with SDK ELFs
- .module_id - .module_id
AccessControlInfo: AccessControlInfo:
#UseExtSaveData : true # UseExtSaveData : true
#ExtSaveDataId: 0xff3ff # ExtSaveDataId: 0xff3ff
#UseExtendedSaveDataAccessControl: true # UseExtendedSaveDataAccessControl: true
#AccessibleSaveDataIds: [0x101, 0x202, 0x303, 0x404, 0x505, 0x606] # AccessibleSaveDataIds: [0x101, 0x202, 0x303, 0x404, 0x505, 0x606]
SystemControlInfo: SystemControlInfo:
SaveDataSize: 128KB SaveDataSize: 128KB
@ -138,7 +139,6 @@ AccessControlInfo:
- $hostio0 - $hostio0
- $hostio1 - $hostio1
- ac:u - ac:u
- am:u
- boss:U - boss:U
- cam:u - cam:u
- cecd:u - cecd:u
@ -162,6 +162,7 @@ AccessControlInfo:
- y2r:u - y2r:u
- ldr:ro - ldr:ro
- ir:USER - ir:USER
- am:u
SystemControlInfo: SystemControlInfo:

View File

@ -1,22 +0,0 @@
#-display text: enter your info between the quotes
#-don't use unicode (i.e. weird characters) or program will barf
#-not certain about string length limits so be conservative
longtitle="FBI CIA Installer"
shortitle="FBI"
publisher="Steveice10"
#-setting flags: don't change unless you know what you're doing
#-1 is on, 0 is off
#-for more info, visit http://3dbrew.org/wiki/SMDH#Flags
visibility =1
autoBoot =0
use3D =1
requireEULA =0
autoSaveOnExit=0
extendedBanner=0
gameRatings =0
useSaveData =1
recordAppUsage=0
disableSaveBU =0

View File

@ -173,7 +173,6 @@ AccessControlInfo:
- $hostio0 - $hostio0
- $hostio1 - $hostio1
- ac:u - ac:u
- am:u
- boss:U - boss:U
- cam:u - cam:u
- cecd:u - cecd:u
@ -199,6 +198,7 @@ AccessControlInfo:
- ir:USER - ir:USER
- ir:u - ir:u
- csnd:SND - csnd:SND
- am:u
SystemControlInfo: SystemControlInfo:

View File

@ -1,2 +0,0 @@
*.bnr
*.icn

View File

@ -1,98 +0,0 @@
from PIL import Image, ImageDraw
import sys,os,platform
f=open('banner.png','rb')
image=Image.open(f)
if image.size[0] != 256 or image.size[1] != 128:
f.close()
print "ERROR: Image must be exactly 256 x 128. Abort."
sys.exit(4)
icon=open('banner.cgfx','wb')
hdr=open("header.bin","rb")
posit=open('map256x128.bin','rb')
'''
chx=[0,1,4,5,16,17,20,21]
chy=[0,2,8,10,32,34,40,42]
w=image.size[0]
h=image.size[1]
'''
n=256*128
i=[0]*(n*2)
cbmdhdr="\x43\x42\x4D\x44\x00\x00\x00\x00\x88"+("\x00"*0x7B)
dump=list(image.getdata())
pos=0
for x in range(n):
#xx=x%w
#yy=x/w
#print xx,yy
#pos=(chx[x%8]+chy[(x>>3)%8])+((x>>6)<<6) fail (i'm not a math major :p)
#print pos
p1=ord(posit.read(1))
p2=ord(posit.read(1))
pos=p1+(p2<<8)
#print p1,p2,pos
r=dump[x][0]>>4
g=dump[x][1]>>4
b=dump[x][2]>>4
a=dump[x][3]>>4
i[pos<<1]= (b<<4) | a
i[(pos<<1)+1]= (r<<4) | g
buf=hdr.read()
hdr.close()
for byte in buf:
icon.write((byte))
for byte in i:
icon.write(chr(byte))
icon.close()
if platform.system() == "Windows":
os.system("DSDecmp.exe -c lz11 banner.cgfx compressed.cgfx")
else:
os.system("wine DSDecmp.exe -c lz11 banner.cgfx compressed.cgfx")
ccgfx=open("compressed.cgfx",'rb')
l=ccgfx.read()
len=len(l)+136
pad=16-(len%16)
l+=("\x00"*pad)
len+=pad
cbmdlen=['']*4
for c in range(4):
cbmdlen[c]=chr(len&255)
len=len>>8
for i in cbmdlen:
cbmdhdr+=i
cbmd=open("banner.cbmd","wb")
cbmdfinal=l
cbmdhdr+=cbmdfinal
cbmd.write(cbmdhdr)
cbmd.close()
print "Done."
f.close()
posit.close()
cbmd.close()
ccgfx.close()
sys.exit(0)

View File

@ -1,59 +0,0 @@
import os,sys
execfile("../../AppData.txt")
icn=open("icon.icn","wb")
bnr=open("banner.bnr","wb")
cptk1=open("icon24/icon24.ctpk","rb")
cptk2=open("icon48/icon48.ctpk","rb")
bcwav1=open("audio/audio.bcwav","rb")
cbmd1=open("banner/banner.cbmd","rb")
ctp1=cptk1.read()
ctp2=cptk2.read()
bcwav=bcwav1.read()
cbmd=cbmd1.read()
header=list("\x53\x4D\x44\x48\x00\x00\x00\x00")
header+=("\x00"*0x1FF8)
header+="\x00\x00\x00\x00\x00\x00\x00\x00\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x00\x00\x00\x00\xFF\xFF\xFF\x7F\x31\x48\x62\x64\x99\x99\x99\x19\x91\x18\x62\x64\xA5\x01\x00\x00\x00\x01\x00\x00\x00\x00\x80\x3F\x32\x41\x79\x24\x00\x00\x00\x00\x00\x00\x00\x00"
header[0x2028]=chr(visibility | autoBoot<<1 | use3D<<2 | requireEULA<<3 | autoSaveOnExit<<4 | extendedBanner<<5 | gameRatings<<6 | useSaveData<<7)
header[0x2029]=chr(recordAppUsage | disableSaveBU<<2)
offset=8
pos=0
for x in range(11):
for c in longtitle:
header[offset+pos*2]=longtitle[pos]
pos+=1
pos=0
offset+=0x80
for c in shortitle:
header[offset+pos*2]=shortitle[pos]
pos+=1
pos=0
offset+=0x100
for c in publisher:
header[offset+pos*2]=publisher[pos]
pos+=1
pos=0
offset+=0x80
header=''.join(header)
header+=(ctp1+ctp2)
icn.write(header)
bnr.write(cbmd+bcwav)
print "banner.bnr built."
print "icon.icn built."
print "Done."
bnr.close()
icn.close()
cptk1.close()
cptk2.close()
bcwav1.close()
cbmd1.close()

View File

@ -1,65 +0,0 @@
from PIL import Image, ImageDraw
import sys,os
icon=open('icon24.ctpk','wb')
f=open('icon24.png','rb')
image=Image.open(f)
posit=open('map24x24.bin','rb')
chx=[0,1,4,5,16,17,20,21]
chy=[0,2,8,10,32,34,40,42]
w=image.size[0]
h=image.size[1]
n=w*w
i=[0]*(n*2)
if w != h:
print "width, height unequal"
f.close()
icon.close()
sys.exit(1)
if w != 24:
print "sides need to be 24 pixels"
f.close()
icon.close()
sys.exit(1)
dump=list(image.getdata())
pos=0
for x in xrange(n):
#xx=x%w
#yy=x/w
#print xx,yy
#pos=(chx[x%8]+chy[(x>>3)%8])+((x>>6)<<6)
#print pos
p1=ord(posit.read(1))
p2=ord(posit.read(1))
pos=p1+(p2<<8)
#print p1,p2,pos
r=dump[x][0]>>3
g=dump[x][1]>>2
b=dump[x][2]>>3
i[pos<<1]=(g&7)<<5| b
i[(pos<<1)+1]=(r)<<3 | g>>3
for byte in i:
icon.write(chr(byte))
print "Done."
icon.close()
f.close()
posit.close()
exit()

View File

@ -1,65 +0,0 @@
from PIL import Image, ImageDraw
import sys,os
icon=open('icon48.ctpk','wb')
f=open('icon48.png','rb')
image=Image.open(f)
posit=open('map48x48.bin','rb')
chx=[0,1,4,5,16,17,20,21]
chy=[0,2,8,10,32,34,40,42]
w=image.size[0]
h=image.size[1]
n=w*w
i=[0]*(n*2)
if w != h:
print "width, height unequal"
f.close()
icon.close()
sys.exit(1)
if w != 48:
print "sides need to be 48 pixels"
f.close()
icon.close()
sys.exit(1)
dump=list(image.getdata())
pos=0
for x in xrange(n):
#xx=x%w
#yy=x/w
#print xx,yy
#pos=(chx[x%8]+chy[(x>>3)%8])+((x>>6)<<6)
#print pos
p1=ord(posit.read(1))
p2=ord(posit.read(1))
pos=p1+(p2<<8)
#print p1,p2,pos
r=dump[x][0]>>3
g=dump[x][1]>>2
b=dump[x][2]>>3
i[pos<<1]=(g&7)<<5| b
i[(pos<<1)+1]=(r)<<3 | g>>3
for byte in i:
icon.write(chr(byte))
print "Done."
icon.close()
f.close()
posit.close()
exit()

View File

@ -547,13 +547,26 @@ App* app_list(MediaType mediaType, u32* count) {
} }
u32 titleCount; u32 titleCount;
AM_GetTitleCount(app_mediatype_to_byte(mediaType), &titleCount); if(AM_GetTitleCount(app_mediatype_to_byte(mediaType), &titleCount) != 0) {
if(count != NULL) {
*count = 0;
}
return (App*) malloc(0);
}
if(count != NULL) { if(count != NULL) {
*count = titleCount; *count = titleCount;
} }
u64 titleIds[titleCount]; u64 titleIds[titleCount];
AM_GetTitleList(app_mediatype_to_byte(mediaType), titleCount, titleIds); if(AM_GetTitleList(app_mediatype_to_byte(mediaType), titleCount, titleIds) != 0) {
if(count != NULL) {
*count = 0;
}
return (App*) malloc(0);
}
App* titles = (App*) malloc(titleCount * sizeof(App)); App* titles = (App*) malloc(titleCount * sizeof(App));
for(int i = 0; i < titleCount; i++) { for(int i = 0; i < titleCount; i++) {
@ -581,23 +594,24 @@ bool app_install(MediaType mediaType, const char* path, bool (*onProgress)(int p
return false; return false;
} }
FILE* fd = fopen(path, "r");
if(!fd) {
return false;
}
fseek(fd, 0, SEEK_END);
u64 size = (u64) ftell(fd);
fseek(fd, 0, SEEK_SET);
if(onProgress != NULL) { if(onProgress != NULL) {
onProgress(0); onProgress(0);
} }
FS_archive sdmcArchive = (FS_archive) {ARCH_SDMC, (FS_path) {PATH_EMPTY, 1, (u8*) ""}}; Handle ciaHandle;
FSUSER_OpenArchive(NULL, &sdmcArchive); if(AM_StartCiaInstall(app_mediatype_to_byte(mediaType), &ciaHandle) != 0) {
Handle fileHandle;
u64 size;
if(FSUSER_OpenFile(NULL, &fileHandle, sdmcArchive, FS_makePath(PATH_CHAR, path + 5), FS_OPEN_READ, FS_ATTRIBUTE_NONE) != 0) {
return false; return false;
} }
FSFILE_GetSize(fileHandle, &size);
Handle ciaHandle;
AM_StartCiaInstall(app_mediatype_to_byte(mediaType), &ciaHandle);
FSFILE_SetSize(ciaHandle, size); FSFILE_SetSize(ciaHandle, size);
u32 bufSize = 1024 * 256; // 256KB u32 bufSize = 1024 * 256; // 256KB
@ -610,23 +624,27 @@ bool app_install(MediaType mediaType, const char* path, bool (*onProgress)(int p
break; break;
} }
u32 bytesRead; u32 bytesRead = fread(buf, 1, bufSize, fd);
FSFILE_Read(fileHandle, &bytesRead, pos, buf, bufSize);
FSFILE_Write(ciaHandle, NULL, pos, buf, bytesRead, FS_WRITE_NOFLUSH); FSFILE_Write(ciaHandle, NULL, pos, buf, bytesRead, FS_WRITE_NOFLUSH);
} }
if(!cancelled) { free(buf);
if(onProgress != NULL) { fclose(fd);
onProgress(100);
}
AM_FinishCiaInstall(app_mediatype_to_byte(mediaType), &ciaHandle); if(cancelled) {
return false;
} }
free(buf); if(onProgress != NULL) {
FSFILE_Close(fileHandle); onProgress(100);
FSUSER_CloseArchive(NULL, &sdmcArchive); }
return !cancelled;
Result res = AM_FinishCiaInstall(app_mediatype_to_byte(mediaType), &ciaHandle);
if(res != 0 && res != 0xC8A044DC) { // Happens when already installed, but seems to have succeeded anyway...
return false;
}
return true;
} }
bool app_delete(MediaType mediaType, App app) { bool app_delete(MediaType mediaType, App app) {
@ -648,22 +666,27 @@ bool app_launch(MediaType mediaType, App app) {
u64 fs_get_free_space(MediaType mediaType) { u64 fs_get_free_space(MediaType mediaType) {
u32 clusterSize; u32 clusterSize;
u32 freeClusters; u32 freeClusters;
Result res = 0;
if(mediaType == NAND) { if(mediaType == NAND) {
FSUSER_GetNandArchiveResource(NULL, NULL, &clusterSize, NULL, &freeClusters); res = FSUSER_GetNandArchiveResource(NULL, NULL, &clusterSize, NULL, &freeClusters);
} else { } else {
FSUSER_GetSdmcArchiveResource(NULL, NULL, &clusterSize, NULL, &freeClusters); res = FSUSER_GetSdmcArchiveResource(NULL, NULL, &clusterSize, NULL, &freeClusters);
}
if(res != 0) {
return 0;
} }
return clusterSize * freeClusters; return clusterSize * freeClusters;
} }
void platform_init() { bool platform_init() {
srvInit(); if(srvInit() != 0 || aptInit() != 0 || hidInit(NULL) != 0 || fsInit() != 0 || sdmcInit() != 0) {
aptInit(); return false;
hidInit(NULL); }
gfxInitDefault(); gfxInitDefault();
fsInit(); return true;
sdmcInit();
} }
void platform_cleanup() { void platform_cleanup() {

View File

@ -113,7 +113,7 @@ bool app_launch(MediaType mediaType, App app);
u64 fs_get_free_space(MediaType mediaType); u64 fs_get_free_space(MediaType mediaType);
void platform_init(); bool platform_init();
void platform_cleanup(); void platform_cleanup();
bool platform_is_running(); bool platform_is_running();
u64 platform_get_time(); u64 platform_get_time();

View File

@ -4,7 +4,9 @@
#include "ui.h" #include "ui.h"
int main(int argc, char **argv) { int main(int argc, char **argv) {
platform_init(); if(!platform_init()) {
return 0;
}
MediaType destination = SD; MediaType destination = SD;
Mode mode = INSTALL; Mode mode = INSTALL;

1
tools/banner/.gitignore vendored Normal file
View File

@ -0,0 +1 @@
__pycache__

255
tools/banner/compress.py Normal file
View File

@ -0,0 +1,255 @@
# used http://code.google.com/p/u-lzss/source/browse/trunk/js/lib/ulzss.js as
# a guide
from sys import stderr
from collections import defaultdict
from operator import itemgetter
from struct import pack, unpack
class SlidingWindow:
# The size of the sliding window
size = 4096
# The minimum displacement.
disp_min = 2
# The hard minimum — a disp less than this can't be represented in the
# compressed stream.
disp_start = 1
# The minimum length for a successful match in the window
match_min = 1
# The maximum length of a successful match, inclusive.
match_max = None
def __init__(self, buf):
self.data = buf
self.hash = defaultdict(list)
self.full = False
self.start = 0
self.stop = 0
#self.index = self.disp_min - 1
self.index = 0
assert self.match_max is not None
def next(self):
if self.index < self.disp_start - 1:
self.index += 1
return
if self.full:
olditem = self.data[self.start]
assert self.hash[olditem][0] == self.start
self.hash[olditem].pop(0)
item = self.data[self.stop]
self.hash[item].append(self.stop)
self.stop += 1
self.index += 1
if self.full:
self.start += 1
else:
if self.size <= self.stop:
self.full = True
def advance(self, n=1):
"""Advance the window by n bytes"""
for _ in range(n):
self.next()
def search(self):
match_max = self.match_max
match_min = self.match_min
counts = []
indices = self.hash[self.data[self.index]]
for i in indices:
matchlen = self.match(i, self.index)
if matchlen >= match_min:
disp = self.index - i
#assert self.index - disp >= 0
#assert self.disp_min <= disp < self.size + self.disp_min
if self.disp_min <= disp:
counts.append((matchlen, -disp))
if matchlen >= match_max:
#assert matchlen == match_max
return counts[-1]
if counts:
match = max(counts, key=itemgetter(0))
return match
return None
def match(self, start, bufstart):
size = self.index - start
if size == 0:
return 0
matchlen = 0
it = range(min(len(self.data) - bufstart, self.match_max))
for i in it:
if self.data[start + (i % size)] == self.data[bufstart + i]:
matchlen += 1
else:
break
return matchlen
class NLZ10Window(SlidingWindow):
size = 4096
match_min = 3
match_max = 3 + 0xf
class NLZ11Window(SlidingWindow):
size = 4096
match_min = 3
match_max = 0x111 + 0xFFFF
class NOverlayWindow(NLZ10Window):
disp_min = 3
def _compress(input, windowclass=NLZ10Window):
"""Generates a stream of tokens. Either a byte (int) or a tuple of (count,
displacement)."""
window = windowclass(input)
i = 0
while True:
if len(input) <= i:
break
match = window.search()
if match:
yield match
#if match[1] == -283:
# raise Exception(match, i)
window.advance(match[0])
i += match[0]
else:
yield input[i]
window.next()
i += 1
def packflags(flags):
n = 0
for i in range(8):
n <<= 1
try:
if flags[i]:
n |= 1
except IndexError:
pass
return n
def chunkit(it, n):
buf = []
for x in it:
buf.append(x)
if n <= len(buf):
yield buf
buf = []
if buf:
yield buf
def compress(input, out):
# header
out.write(pack("<L", (len(input) << 8) + 0x10))
# body
length = 0
for tokens in chunkit(_compress(input), 8):
flags = [type(t) == tuple for t in tokens]
out.write(pack(">B", packflags(flags)))
for t in tokens:
if type(t) == tuple:
count, disp = t
count -= 3
disp = (-disp) - 1
assert 0 <= disp < 4096
sh = (count << 12) | disp
out.write(pack(">H", sh))
else:
out.write(pack(">B", t))
length += 1
length += sum(2 if f else 1 for f in flags)
# padding
padding = 4 - (length % 4 or 4)
if padding:
out.write(b'\xff' * padding)
def compress_nlz11(input, out):
# header
out.write(pack("<L", (len(input) << 8) + 0x11))
# body
length = 0
for tokens in chunkit(_compress(input, windowclass=NLZ11Window), 8):
flags = [type(t) == tuple for t in tokens]
out.write(pack(">B", packflags(flags)))
length += 1
for t in tokens:
if type(t) == tuple:
count, disp = t
disp = (-disp) - 1
#if disp == 282:
# raise Exception
assert 0 <= disp <= 0xFFF
if count <= 1 + 0xF:
count -= 1
assert 2 <= count <= 0xF
sh = (count << 12) | disp
out.write(pack(">H", sh))
length += 2
elif count <= 0x11 + 0xFF:
count -= 0x11
assert 0 <= count <= 0xFF
b = count >> 4
sh = ((count & 0xF) << 12) | disp
out.write(pack(">BH", b, sh))
length += 3
elif count <= 0x111 + 0xFFFF:
count -= 0x111
assert 0 <= count <= 0xFFFF
l = (1 << 28) | (count << 12) | disp
out.write(pack(">L", l))
length += 4
else:
raise ValueError(count)
else:
out.write(pack(">B", t))
length += 1
# padding
padding = 4 - (length % 4 or 4)
if padding:
out.write(b'\xff' * padding)
def dump_compress_nlz11(input, out):
# body
length = 0
def dump():
for t in _compress(input, windowclass=NLZ11Window):
if type(t) == tuple:
yield t
from pprint import pprint
pprint(list(dump()))
if __name__ == '__main__':
from sys import stdout, argv
data = open(argv[1], "rb").read()
stdout = stdout.detach()
#compress(data, stdout)
compress_nlz11(data, stdout)
#dump_compress_nlz11(data, stdout)

163
tools/banner/create.py Normal file
View File

@ -0,0 +1,163 @@
from PIL import Image, ImageDraw
from io import BytesIO
from compress import compress_nlz11
from struct import pack
import sys
longtitle=sys.argv[1]
shortitle=sys.argv[2]
publisher=sys.argv[3]
# fixed variables; not likely these will be used in this context, but here just in case.
visibility =1
autoBoot =0
use3D =1
requireEULA =0
autoSaveOnExit=0
extendedBanner=0
gameRatings =0
useSaveData =1
recordAppUsage=0
disableSaveBU =0
def make_icon(file, size):
f=open(file,'rb')
image=Image.open(f)
posit=open('map' + str(size) + 'x' + str(size) + '.bin','rb')
w=image.size[0]
h=image.size[1]
n=w*w
i=[0]*(n*2)
if w != h:
print("width, height unequal")
f.close()
sys.exit(1)
if w != size:
print("sides need to be " + str(size) + " pixels")
f.close()
sys.exit(1)
dump=list(image.getdata())
pos=0
for x in range(n):
p1=ord(posit.read(1))
p2=ord(posit.read(1))
pos=p1+(p2<<8)
r=dump[x][0]>>3
g=dump[x][1]>>2
b=dump[x][2]>>3
i[pos<<1]=(g&7)<<5| b
i[(pos<<1)+1]=(r)<<3 | g>>3
f.close()
posit.close()
return bytearray(i)
def make_banner(file):
f=open(file,'rb')
image=Image.open(f)
if image.size[0] != 256 or image.size[1] != 128:
f.close()
print("ERROR: Image must be exactly 256 x 128. Abort.")
sys.exit(4)
hdr=open("header.bin","rb")
posit=open('map256x128.bin','rb')
n=256*128
i=[0]*(n*2)
cbmdhdr=bytearray(b"\x43\x42\x4D\x44\x00\x00\x00\x00\x88"+(b"\x00"*0x7B))
dump=list(image.getdata())
pos=0
for x in range(n):
p1=ord(posit.read(1))
p2=ord(posit.read(1))
pos=p1+(p2<<8)
r=dump[x][0]>>4
g=dump[x][1]>>4
b=dump[x][2]>>4
a=dump[x][3]>>4
i[pos<<1]= (b<<4) | a
i[(pos<<1)+1]= (r<<4) | g
buf=hdr.read()
hdr.close()
out = BytesIO()
compress_nlz11(buf+bytearray(i), out)
l=bytearray(out.getvalue())
length=len(l)+136
pad=16-(length%16)
l+=bytearray([0]*pad)
length+=pad
for c in range(4):
cbmdhdr+=pack("B", length & 255)
length=length>>8
cbmdhdr+=l
f.close()
posit.close()
return bytearray(cbmdhdr)
def make_audio():
# TODO: Convert WAV file.
bcwav1=open("audio.bcwav","rb")
bcwav=bcwav1.read()
bcwav1.close()
return bytearray(bcwav)
ctp1=make_icon(sys.argv[4], 24)
ctp2=make_icon(sys.argv[5], 48)
bcwav=make_audio()
cbmd=make_banner(sys.argv[6])
icn=open(sys.argv[7],"wb")
bnr=open(sys.argv[8],"wb")
header=bytearray(b"\x53\x4D\x44\x48\x00\x00\x00\x00")+bytearray(b"\x00"*0x1FF8)+bytearray(b"\x00\x00\x00\x00\x00\x00\x00\x00\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x20\x00\x00\x00\x00\xFF\xFF\xFF\x7F\x31\x48\x62\x64\x99\x99\x99\x19\x91\x18\x62\x64\xA5\x01\x00\x00\x00\x01\x00\x00\x00\x00\x80\x3F\x32\x41\x79\x24\x00\x00\x00\x00\x00\x00\x00\x00")
header[0x2028]=(visibility | autoBoot<<1 | use3D<<2 | requireEULA<<3 | autoSaveOnExit<<4 | extendedBanner<<5 | gameRatings<<6 | useSaveData<<7)
header[0x2029]=(recordAppUsage | disableSaveBU<<2)
offset=8
pos=0
for x in range(11):
for c in longtitle:
header[offset+pos*2]=ord(longtitle[pos])
pos+=1
pos=0
offset+=0x80
for c in shortitle:
header[offset+pos*2]=ord(shortitle[pos])
pos+=1
pos=0
offset+=0x100
for c in publisher:
header[offset+pos*2]=ord(publisher[pos])
pos+=1
pos=0
offset+=0x80
header+=(ctp1+ctp2)
icn.write(header)
bnr.write(cbmd+bcwav)
bnr.close()
icn.close()

View File

Before

Width:  |  Height:  |  Size: 1.1 KiB

After

Width:  |  Height:  |  Size: 1.1 KiB

View File

Before

Width:  |  Height:  |  Size: 64 KiB

After

Width:  |  Height:  |  Size: 64 KiB

View File

Before

Width:  |  Height:  |  Size: 4.5 KiB

After

Width:  |  Height:  |  Size: 4.5 KiB