v3.0.0-rc1
94
.clang-format
Normal file
@ -0,0 +1,94 @@
|
|||||||
|
---
|
||||||
|
Language: Cpp
|
||||||
|
# BasedOnStyle: LLVM
|
||||||
|
AccessModifierOffset: -4
|
||||||
|
AlignAfterOpenBracket: DontAlign
|
||||||
|
AlignConsecutiveAssignments: true
|
||||||
|
AlignConsecutiveDeclarations: false
|
||||||
|
AlignEscapedNewlinesLeft: false
|
||||||
|
AlignOperands: true
|
||||||
|
AlignTrailingComments: true
|
||||||
|
AllowAllParametersOfDeclarationOnNextLine: false
|
||||||
|
AllowShortBlocksOnASingleLine: false
|
||||||
|
AllowShortCaseLabelsOnASingleLine: false
|
||||||
|
AllowShortFunctionsOnASingleLine: None
|
||||||
|
AllowShortIfStatementsOnASingleLine: false
|
||||||
|
AllowShortLoopsOnASingleLine: false
|
||||||
|
#AlwaysBreakAfterDefinitionReturnType: None
|
||||||
|
AlwaysBreakAfterReturnType: None
|
||||||
|
AlwaysBreakBeforeMultilineStrings: false
|
||||||
|
AlwaysBreakTemplateDeclarations: true
|
||||||
|
BinPackArguments: false
|
||||||
|
BinPackParameters: false
|
||||||
|
BraceWrapping:
|
||||||
|
AfterClass: true
|
||||||
|
AfterControlStatement: true
|
||||||
|
AfterEnum: true
|
||||||
|
AfterFunction: true
|
||||||
|
AfterNamespace: true
|
||||||
|
AfterStruct: true
|
||||||
|
AfterUnion: true
|
||||||
|
BeforeCatch: true
|
||||||
|
BeforeElse: true
|
||||||
|
IndentBraces: false
|
||||||
|
# SplitEmptyFunction: true
|
||||||
|
# SplitEmptyRecord: true
|
||||||
|
# SplitEmptyNamespace: true
|
||||||
|
BreakBeforeBinaryOperators: None
|
||||||
|
BreakBeforeBraces: Custom
|
||||||
|
BreakBeforeTernaryOperators: true
|
||||||
|
#BreakConstructorInitializers: AfterColon
|
||||||
|
BreakConstructorInitializersBeforeComma: false
|
||||||
|
BreakStringLiterals: true
|
||||||
|
ColumnLimit: 100
|
||||||
|
CommentPragmas: '^ IWYU pragma:'
|
||||||
|
ConstructorInitializerAllOnOneLineOrOnePerLine: true
|
||||||
|
ConstructorInitializerIndentWidth: 4
|
||||||
|
ContinuationIndentWidth: 4
|
||||||
|
Cpp11BracedListStyle: true
|
||||||
|
DerivePointerAlignment: false
|
||||||
|
DisableFormat: false
|
||||||
|
ExperimentalAutoDetectBinPacking: false
|
||||||
|
FixNamespaceComments: false
|
||||||
|
ForEachMacros: [ foreach ]
|
||||||
|
IncludeCategories:
|
||||||
|
- Regex: '^"(llvm|llvm-c|clang|clang-c)/'
|
||||||
|
Priority: 2
|
||||||
|
- Regex: '^(<|"(gtest|isl|json)/)'
|
||||||
|
Priority: 3
|
||||||
|
- Regex: '.*'
|
||||||
|
Priority: 1
|
||||||
|
IncludeIsMainRegex: '$'
|
||||||
|
IndentCaseLabels: false
|
||||||
|
IndentWidth: 4
|
||||||
|
IndentWrappedFunctionNames: true
|
||||||
|
KeepEmptyLinesAtTheStartOfBlocks: false
|
||||||
|
MacroBlockBegin: ''
|
||||||
|
MacroBlockEnd: ''
|
||||||
|
MaxEmptyLinesToKeep: 1
|
||||||
|
NamespaceIndentation: None
|
||||||
|
PenaltyBreakBeforeFirstCallParameter: 19
|
||||||
|
PenaltyBreakComment: 300
|
||||||
|
PenaltyBreakFirstLessLess: 120
|
||||||
|
PenaltyBreakString: 1000
|
||||||
|
PenaltyExcessCharacter: 1000000
|
||||||
|
PenaltyReturnTypeOnItsOwnLine: 60
|
||||||
|
PointerAlignment: Right
|
||||||
|
ReflowComments: true
|
||||||
|
SortIncludes: true
|
||||||
|
SpaceAfterCStyleCast: false
|
||||||
|
SpaceAfterTemplateKeyword: true
|
||||||
|
SpaceBeforeAssignmentOperators: true
|
||||||
|
SpaceBeforeParens: Always
|
||||||
|
SpaceInEmptyParentheses: false
|
||||||
|
SpacesBeforeTrailingComments: 1
|
||||||
|
SpacesInAngles: false
|
||||||
|
SpacesInContainerLiterals: true
|
||||||
|
SpacesInCStyleCastParentheses: false
|
||||||
|
SpacesInParentheses: false
|
||||||
|
SpacesInSquareBrackets: false
|
||||||
|
Standard: Cpp11
|
||||||
|
TabWidth: 4
|
||||||
|
UseTab: ForIndentation
|
||||||
|
...
|
||||||
|
|
4
.gitignore
vendored
@ -9,5 +9,9 @@
|
|||||||
*.nso
|
*.nso
|
||||||
*.pfs0
|
*.pfs0
|
||||||
*.smdh
|
*.smdh
|
||||||
|
.gdb_history
|
||||||
build.*/
|
build.*/
|
||||||
ftpd
|
ftpd
|
||||||
|
romfs.3ds/*.t3x
|
||||||
|
romfs.switch/*.zst
|
||||||
|
romfs.switch/shaders/*.dksh
|
||||||
|
689
LICENSE
@ -1,25 +1,674 @@
|
|||||||
This is free and unencumbered software released into the public domain.
|
GNU GENERAL PUBLIC LICENSE
|
||||||
|
Version 3, 29 June 2007
|
||||||
|
|
||||||
Anyone is free to copy, modify, publish, use, compile, sell, or
|
Copyright (C) 2007 Free Software Foundation, Inc. <https://fsf.org/>
|
||||||
distribute this software, either in source code form or as a compiled
|
Everyone is permitted to copy and distribute verbatim copies
|
||||||
binary, for any purpose, commercial or non-commercial, and by any
|
of this license document, but changing it is not allowed.
|
||||||
means.
|
|
||||||
|
|
||||||
In jurisdictions that recognize copyright laws, the author or authors
|
Preamble
|
||||||
of this software dedicate any and all copyright interest in the
|
|
||||||
software to the public domain. We make this dedication for the benefit
|
|
||||||
of the public at large and to the detriment of our heirs and
|
|
||||||
successors. We intend this dedication to be an overt act of
|
|
||||||
relinquishment in perpetuity of all present and future rights to this
|
|
||||||
software under copyright law.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
The GNU General Public License is a free, copyleft license for
|
||||||
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
software and other kinds of works.
|
||||||
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
|
|
||||||
IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR
|
|
||||||
OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
|
|
||||||
ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
|
|
||||||
OTHER DEALINGS IN THE SOFTWARE.
|
|
||||||
|
|
||||||
For more information, please refer to <http://unlicense.org>
|
The licenses for most software and other practical works are designed
|
||||||
|
to take away your freedom to share and change the works. By contrast,
|
||||||
|
the GNU General Public License is intended to guarantee your freedom to
|
||||||
|
share and change all versions of a program--to make sure it remains free
|
||||||
|
software for all its users. We, the Free Software Foundation, use the
|
||||||
|
GNU General Public License for most of our software; it applies also to
|
||||||
|
any other work released this way by its authors. You can apply it to
|
||||||
|
your programs, too.
|
||||||
|
|
||||||
|
When we speak of free software, we are referring to freedom, not
|
||||||
|
price. Our General Public Licenses are designed to make sure that you
|
||||||
|
have the freedom to distribute copies of free software (and charge for
|
||||||
|
them if you wish), that you receive source code or can get it if you
|
||||||
|
want it, that you can change the software or use pieces of it in new
|
||||||
|
free programs, and that you know you can do these things.
|
||||||
|
|
||||||
|
To protect your rights, we need to prevent others from denying you
|
||||||
|
these rights or asking you to surrender the rights. Therefore, you have
|
||||||
|
certain responsibilities if you distribute copies of the software, or if
|
||||||
|
you modify it: responsibilities to respect the freedom of others.
|
||||||
|
|
||||||
|
For example, if you distribute copies of such a program, whether
|
||||||
|
gratis or for a fee, you must pass on to the recipients the same
|
||||||
|
freedoms that you received. You must make sure that they, too, receive
|
||||||
|
or can get the source code. And you must show them these terms so they
|
||||||
|
know their rights.
|
||||||
|
|
||||||
|
Developers that use the GNU GPL protect your rights with two steps:
|
||||||
|
(1) assert copyright on the software, and (2) offer you this License
|
||||||
|
giving you legal permission to copy, distribute and/or modify it.
|
||||||
|
|
||||||
|
For the developers' and authors' protection, the GPL clearly explains
|
||||||
|
that there is no warranty for this free software. For both users' and
|
||||||
|
authors' sake, the GPL requires that modified versions be marked as
|
||||||
|
changed, so that their problems will not be attributed erroneously to
|
||||||
|
authors of previous versions.
|
||||||
|
|
||||||
|
Some devices are designed to deny users access to install or run
|
||||||
|
modified versions of the software inside them, although the manufacturer
|
||||||
|
can do so. This is fundamentally incompatible with the aim of
|
||||||
|
protecting users' freedom to change the software. The systematic
|
||||||
|
pattern of such abuse occurs in the area of products for individuals to
|
||||||
|
use, which is precisely where it is most unacceptable. Therefore, we
|
||||||
|
have designed this version of the GPL to prohibit the practice for those
|
||||||
|
products. If such problems arise substantially in other domains, we
|
||||||
|
stand ready to extend this provision to those domains in future versions
|
||||||
|
of the GPL, as needed to protect the freedom of users.
|
||||||
|
|
||||||
|
Finally, every program is threatened constantly by software patents.
|
||||||
|
States should not allow patents to restrict development and use of
|
||||||
|
software on general-purpose computers, but in those that do, we wish to
|
||||||
|
avoid the special danger that patents applied to a free program could
|
||||||
|
make it effectively proprietary. To prevent this, the GPL assures that
|
||||||
|
patents cannot be used to render the program non-free.
|
||||||
|
|
||||||
|
The precise terms and conditions for copying, distribution and
|
||||||
|
modification follow.
|
||||||
|
|
||||||
|
TERMS AND CONDITIONS
|
||||||
|
|
||||||
|
0. Definitions.
|
||||||
|
|
||||||
|
"This License" refers to version 3 of the GNU General Public License.
|
||||||
|
|
||||||
|
"Copyright" also means copyright-like laws that apply to other kinds of
|
||||||
|
works, such as semiconductor masks.
|
||||||
|
|
||||||
|
"The Program" refers to any copyrightable work licensed under this
|
||||||
|
License. Each licensee is addressed as "you". "Licensees" and
|
||||||
|
"recipients" may be individuals or organizations.
|
||||||
|
|
||||||
|
To "modify" a work means to copy from or adapt all or part of the work
|
||||||
|
in a fashion requiring copyright permission, other than the making of an
|
||||||
|
exact copy. The resulting work is called a "modified version" of the
|
||||||
|
earlier work or a work "based on" the earlier work.
|
||||||
|
|
||||||
|
A "covered work" means either the unmodified Program or a work based
|
||||||
|
on the Program.
|
||||||
|
|
||||||
|
To "propagate" a work means to do anything with it that, without
|
||||||
|
permission, would make you directly or secondarily liable for
|
||||||
|
infringement under applicable copyright law, except executing it on a
|
||||||
|
computer or modifying a private copy. Propagation includes copying,
|
||||||
|
distribution (with or without modification), making available to the
|
||||||
|
public, and in some countries other activities as well.
|
||||||
|
|
||||||
|
To "convey" a work means any kind of propagation that enables other
|
||||||
|
parties to make or receive copies. Mere interaction with a user through
|
||||||
|
a computer network, with no transfer of a copy, is not conveying.
|
||||||
|
|
||||||
|
An interactive user interface displays "Appropriate Legal Notices"
|
||||||
|
to the extent that it includes a convenient and prominently visible
|
||||||
|
feature that (1) displays an appropriate copyright notice, and (2)
|
||||||
|
tells the user that there is no warranty for the work (except to the
|
||||||
|
extent that warranties are provided), that licensees may convey the
|
||||||
|
work under this License, and how to view a copy of this License. If
|
||||||
|
the interface presents a list of user commands or options, such as a
|
||||||
|
menu, a prominent item in the list meets this criterion.
|
||||||
|
|
||||||
|
1. Source Code.
|
||||||
|
|
||||||
|
The "source code" for a work means the preferred form of the work
|
||||||
|
for making modifications to it. "Object code" means any non-source
|
||||||
|
form of a work.
|
||||||
|
|
||||||
|
A "Standard Interface" means an interface that either is an official
|
||||||
|
standard defined by a recognized standards body, or, in the case of
|
||||||
|
interfaces specified for a particular programming language, one that
|
||||||
|
is widely used among developers working in that language.
|
||||||
|
|
||||||
|
The "System Libraries" of an executable work include anything, other
|
||||||
|
than the work as a whole, that (a) is included in the normal form of
|
||||||
|
packaging a Major Component, but which is not part of that Major
|
||||||
|
Component, and (b) serves only to enable use of the work with that
|
||||||
|
Major Component, or to implement a Standard Interface for which an
|
||||||
|
implementation is available to the public in source code form. A
|
||||||
|
"Major Component", in this context, means a major essential component
|
||||||
|
(kernel, window system, and so on) of the specific operating system
|
||||||
|
(if any) on which the executable work runs, or a compiler used to
|
||||||
|
produce the work, or an object code interpreter used to run it.
|
||||||
|
|
||||||
|
The "Corresponding Source" for a work in object code form means all
|
||||||
|
the source code needed to generate, install, and (for an executable
|
||||||
|
work) run the object code and to modify the work, including scripts to
|
||||||
|
control those activities. However, it does not include the work's
|
||||||
|
System Libraries, or general-purpose tools or generally available free
|
||||||
|
programs which are used unmodified in performing those activities but
|
||||||
|
which are not part of the work. For example, Corresponding Source
|
||||||
|
includes interface definition files associated with source files for
|
||||||
|
the work, and the source code for shared libraries and dynamically
|
||||||
|
linked subprograms that the work is specifically designed to require,
|
||||||
|
such as by intimate data communication or control flow between those
|
||||||
|
subprograms and other parts of the work.
|
||||||
|
|
||||||
|
The Corresponding Source need not include anything that users
|
||||||
|
can regenerate automatically from other parts of the Corresponding
|
||||||
|
Source.
|
||||||
|
|
||||||
|
The Corresponding Source for a work in source code form is that
|
||||||
|
same work.
|
||||||
|
|
||||||
|
2. Basic Permissions.
|
||||||
|
|
||||||
|
All rights granted under this License are granted for the term of
|
||||||
|
copyright on the Program, and are irrevocable provided the stated
|
||||||
|
conditions are met. This License explicitly affirms your unlimited
|
||||||
|
permission to run the unmodified Program. The output from running a
|
||||||
|
covered work is covered by this License only if the output, given its
|
||||||
|
content, constitutes a covered work. This License acknowledges your
|
||||||
|
rights of fair use or other equivalent, as provided by copyright law.
|
||||||
|
|
||||||
|
You may make, run and propagate covered works that you do not
|
||||||
|
convey, without conditions so long as your license otherwise remains
|
||||||
|
in force. You may convey covered works to others for the sole purpose
|
||||||
|
of having them make modifications exclusively for you, or provide you
|
||||||
|
with facilities for running those works, provided that you comply with
|
||||||
|
the terms of this License in conveying all material for which you do
|
||||||
|
not control copyright. Those thus making or running the covered works
|
||||||
|
for you must do so exclusively on your behalf, under your direction
|
||||||
|
and control, on terms that prohibit them from making any copies of
|
||||||
|
your copyrighted material outside their relationship with you.
|
||||||
|
|
||||||
|
Conveying under any other circumstances is permitted solely under
|
||||||
|
the conditions stated below. Sublicensing is not allowed; section 10
|
||||||
|
makes it unnecessary.
|
||||||
|
|
||||||
|
3. Protecting Users' Legal Rights From Anti-Circumvention Law.
|
||||||
|
|
||||||
|
No covered work shall be deemed part of an effective technological
|
||||||
|
measure under any applicable law fulfilling obligations under article
|
||||||
|
11 of the WIPO copyright treaty adopted on 20 December 1996, or
|
||||||
|
similar laws prohibiting or restricting circumvention of such
|
||||||
|
measures.
|
||||||
|
|
||||||
|
When you convey a covered work, you waive any legal power to forbid
|
||||||
|
circumvention of technological measures to the extent such circumvention
|
||||||
|
is effected by exercising rights under this License with respect to
|
||||||
|
the covered work, and you disclaim any intention to limit operation or
|
||||||
|
modification of the work as a means of enforcing, against the work's
|
||||||
|
users, your or third parties' legal rights to forbid circumvention of
|
||||||
|
technological measures.
|
||||||
|
|
||||||
|
4. Conveying Verbatim Copies.
|
||||||
|
|
||||||
|
You may convey verbatim copies of the Program's source code as you
|
||||||
|
receive it, in any medium, provided that you conspicuously and
|
||||||
|
appropriately publish on each copy an appropriate copyright notice;
|
||||||
|
keep intact all notices stating that this License and any
|
||||||
|
non-permissive terms added in accord with section 7 apply to the code;
|
||||||
|
keep intact all notices of the absence of any warranty; and give all
|
||||||
|
recipients a copy of this License along with the Program.
|
||||||
|
|
||||||
|
You may charge any price or no price for each copy that you convey,
|
||||||
|
and you may offer support or warranty protection for a fee.
|
||||||
|
|
||||||
|
5. Conveying Modified Source Versions.
|
||||||
|
|
||||||
|
You may convey a work based on the Program, or the modifications to
|
||||||
|
produce it from the Program, in the form of source code under the
|
||||||
|
terms of section 4, provided that you also meet all of these conditions:
|
||||||
|
|
||||||
|
a) The work must carry prominent notices stating that you modified
|
||||||
|
it, and giving a relevant date.
|
||||||
|
|
||||||
|
b) The work must carry prominent notices stating that it is
|
||||||
|
released under this License and any conditions added under section
|
||||||
|
7. This requirement modifies the requirement in section 4 to
|
||||||
|
"keep intact all notices".
|
||||||
|
|
||||||
|
c) You must license the entire work, as a whole, under this
|
||||||
|
License to anyone who comes into possession of a copy. This
|
||||||
|
License will therefore apply, along with any applicable section 7
|
||||||
|
additional terms, to the whole of the work, and all its parts,
|
||||||
|
regardless of how they are packaged. This License gives no
|
||||||
|
permission to license the work in any other way, but it does not
|
||||||
|
invalidate such permission if you have separately received it.
|
||||||
|
|
||||||
|
d) If the work has interactive user interfaces, each must display
|
||||||
|
Appropriate Legal Notices; however, if the Program has interactive
|
||||||
|
interfaces that do not display Appropriate Legal Notices, your
|
||||||
|
work need not make them do so.
|
||||||
|
|
||||||
|
A compilation of a covered work with other separate and independent
|
||||||
|
works, which are not by their nature extensions of the covered work,
|
||||||
|
and which are not combined with it such as to form a larger program,
|
||||||
|
in or on a volume of a storage or distribution medium, is called an
|
||||||
|
"aggregate" if the compilation and its resulting copyright are not
|
||||||
|
used to limit the access or legal rights of the compilation's users
|
||||||
|
beyond what the individual works permit. Inclusion of a covered work
|
||||||
|
in an aggregate does not cause this License to apply to the other
|
||||||
|
parts of the aggregate.
|
||||||
|
|
||||||
|
6. Conveying Non-Source Forms.
|
||||||
|
|
||||||
|
You may convey a covered work in object code form under the terms
|
||||||
|
of sections 4 and 5, provided that you also convey the
|
||||||
|
machine-readable Corresponding Source under the terms of this License,
|
||||||
|
in one of these ways:
|
||||||
|
|
||||||
|
a) Convey the object code in, or embodied in, a physical product
|
||||||
|
(including a physical distribution medium), accompanied by the
|
||||||
|
Corresponding Source fixed on a durable physical medium
|
||||||
|
customarily used for software interchange.
|
||||||
|
|
||||||
|
b) Convey the object code in, or embodied in, a physical product
|
||||||
|
(including a physical distribution medium), accompanied by a
|
||||||
|
written offer, valid for at least three years and valid for as
|
||||||
|
long as you offer spare parts or customer support for that product
|
||||||
|
model, to give anyone who possesses the object code either (1) a
|
||||||
|
copy of the Corresponding Source for all the software in the
|
||||||
|
product that is covered by this License, on a durable physical
|
||||||
|
medium customarily used for software interchange, for a price no
|
||||||
|
more than your reasonable cost of physically performing this
|
||||||
|
conveying of source, or (2) access to copy the
|
||||||
|
Corresponding Source from a network server at no charge.
|
||||||
|
|
||||||
|
c) Convey individual copies of the object code with a copy of the
|
||||||
|
written offer to provide the Corresponding Source. This
|
||||||
|
alternative is allowed only occasionally and noncommercially, and
|
||||||
|
only if you received the object code with such an offer, in accord
|
||||||
|
with subsection 6b.
|
||||||
|
|
||||||
|
d) Convey the object code by offering access from a designated
|
||||||
|
place (gratis or for a charge), and offer equivalent access to the
|
||||||
|
Corresponding Source in the same way through the same place at no
|
||||||
|
further charge. You need not require recipients to copy the
|
||||||
|
Corresponding Source along with the object code. If the place to
|
||||||
|
copy the object code is a network server, the Corresponding Source
|
||||||
|
may be on a different server (operated by you or a third party)
|
||||||
|
that supports equivalent copying facilities, provided you maintain
|
||||||
|
clear directions next to the object code saying where to find the
|
||||||
|
Corresponding Source. Regardless of what server hosts the
|
||||||
|
Corresponding Source, you remain obligated to ensure that it is
|
||||||
|
available for as long as needed to satisfy these requirements.
|
||||||
|
|
||||||
|
e) Convey the object code using peer-to-peer transmission, provided
|
||||||
|
you inform other peers where the object code and Corresponding
|
||||||
|
Source of the work are being offered to the general public at no
|
||||||
|
charge under subsection 6d.
|
||||||
|
|
||||||
|
A separable portion of the object code, whose source code is excluded
|
||||||
|
from the Corresponding Source as a System Library, need not be
|
||||||
|
included in conveying the object code work.
|
||||||
|
|
||||||
|
A "User Product" is either (1) a "consumer product", which means any
|
||||||
|
tangible personal property which is normally used for personal, family,
|
||||||
|
or household purposes, or (2) anything designed or sold for incorporation
|
||||||
|
into a dwelling. In determining whether a product is a consumer product,
|
||||||
|
doubtful cases shall be resolved in favor of coverage. For a particular
|
||||||
|
product received by a particular user, "normally used" refers to a
|
||||||
|
typical or common use of that class of product, regardless of the status
|
||||||
|
of the particular user or of the way in which the particular user
|
||||||
|
actually uses, or expects or is expected to use, the product. A product
|
||||||
|
is a consumer product regardless of whether the product has substantial
|
||||||
|
commercial, industrial or non-consumer uses, unless such uses represent
|
||||||
|
the only significant mode of use of the product.
|
||||||
|
|
||||||
|
"Installation Information" for a User Product means any methods,
|
||||||
|
procedures, authorization keys, or other information required to install
|
||||||
|
and execute modified versions of a covered work in that User Product from
|
||||||
|
a modified version of its Corresponding Source. The information must
|
||||||
|
suffice to ensure that the continued functioning of the modified object
|
||||||
|
code is in no case prevented or interfered with solely because
|
||||||
|
modification has been made.
|
||||||
|
|
||||||
|
If you convey an object code work under this section in, or with, or
|
||||||
|
specifically for use in, a User Product, and the conveying occurs as
|
||||||
|
part of a transaction in which the right of possession and use of the
|
||||||
|
User Product is transferred to the recipient in perpetuity or for a
|
||||||
|
fixed term (regardless of how the transaction is characterized), the
|
||||||
|
Corresponding Source conveyed under this section must be accompanied
|
||||||
|
by the Installation Information. But this requirement does not apply
|
||||||
|
if neither you nor any third party retains the ability to install
|
||||||
|
modified object code on the User Product (for example, the work has
|
||||||
|
been installed in ROM).
|
||||||
|
|
||||||
|
The requirement to provide Installation Information does not include a
|
||||||
|
requirement to continue to provide support service, warranty, or updates
|
||||||
|
for a work that has been modified or installed by the recipient, or for
|
||||||
|
the User Product in which it has been modified or installed. Access to a
|
||||||
|
network may be denied when the modification itself materially and
|
||||||
|
adversely affects the operation of the network or violates the rules and
|
||||||
|
protocols for communication across the network.
|
||||||
|
|
||||||
|
Corresponding Source conveyed, and Installation Information provided,
|
||||||
|
in accord with this section must be in a format that is publicly
|
||||||
|
documented (and with an implementation available to the public in
|
||||||
|
source code form), and must require no special password or key for
|
||||||
|
unpacking, reading or copying.
|
||||||
|
|
||||||
|
7. Additional Terms.
|
||||||
|
|
||||||
|
"Additional permissions" are terms that supplement the terms of this
|
||||||
|
License by making exceptions from one or more of its conditions.
|
||||||
|
Additional permissions that are applicable to the entire Program shall
|
||||||
|
be treated as though they were included in this License, to the extent
|
||||||
|
that they are valid under applicable law. If additional permissions
|
||||||
|
apply only to part of the Program, that part may be used separately
|
||||||
|
under those permissions, but the entire Program remains governed by
|
||||||
|
this License without regard to the additional permissions.
|
||||||
|
|
||||||
|
When you convey a copy of a covered work, you may at your option
|
||||||
|
remove any additional permissions from that copy, or from any part of
|
||||||
|
it. (Additional permissions may be written to require their own
|
||||||
|
removal in certain cases when you modify the work.) You may place
|
||||||
|
additional permissions on material, added by you to a covered work,
|
||||||
|
for which you have or can give appropriate copyright permission.
|
||||||
|
|
||||||
|
Notwithstanding any other provision of this License, for material you
|
||||||
|
add to a covered work, you may (if authorized by the copyright holders of
|
||||||
|
that material) supplement the terms of this License with terms:
|
||||||
|
|
||||||
|
a) Disclaiming warranty or limiting liability differently from the
|
||||||
|
terms of sections 15 and 16 of this License; or
|
||||||
|
|
||||||
|
b) Requiring preservation of specified reasonable legal notices or
|
||||||
|
author attributions in that material or in the Appropriate Legal
|
||||||
|
Notices displayed by works containing it; or
|
||||||
|
|
||||||
|
c) Prohibiting misrepresentation of the origin of that material, or
|
||||||
|
requiring that modified versions of such material be marked in
|
||||||
|
reasonable ways as different from the original version; or
|
||||||
|
|
||||||
|
d) Limiting the use for publicity purposes of names of licensors or
|
||||||
|
authors of the material; or
|
||||||
|
|
||||||
|
e) Declining to grant rights under trademark law for use of some
|
||||||
|
trade names, trademarks, or service marks; or
|
||||||
|
|
||||||
|
f) Requiring indemnification of licensors and authors of that
|
||||||
|
material by anyone who conveys the material (or modified versions of
|
||||||
|
it) with contractual assumptions of liability to the recipient, for
|
||||||
|
any liability that these contractual assumptions directly impose on
|
||||||
|
those licensors and authors.
|
||||||
|
|
||||||
|
All other non-permissive additional terms are considered "further
|
||||||
|
restrictions" within the meaning of section 10. If the Program as you
|
||||||
|
received it, or any part of it, contains a notice stating that it is
|
||||||
|
governed by this License along with a term that is a further
|
||||||
|
restriction, you may remove that term. If a license document contains
|
||||||
|
a further restriction but permits relicensing or conveying under this
|
||||||
|
License, you may add to a covered work material governed by the terms
|
||||||
|
of that license document, provided that the further restriction does
|
||||||
|
not survive such relicensing or conveying.
|
||||||
|
|
||||||
|
If you add terms to a covered work in accord with this section, you
|
||||||
|
must place, in the relevant source files, a statement of the
|
||||||
|
additional terms that apply to those files, or a notice indicating
|
||||||
|
where to find the applicable terms.
|
||||||
|
|
||||||
|
Additional terms, permissive or non-permissive, may be stated in the
|
||||||
|
form of a separately written license, or stated as exceptions;
|
||||||
|
the above requirements apply either way.
|
||||||
|
|
||||||
|
8. Termination.
|
||||||
|
|
||||||
|
You may not propagate or modify a covered work except as expressly
|
||||||
|
provided under this License. Any attempt otherwise to propagate or
|
||||||
|
modify it is void, and will automatically terminate your rights under
|
||||||
|
this License (including any patent licenses granted under the third
|
||||||
|
paragraph of section 11).
|
||||||
|
|
||||||
|
However, if you cease all violation of this License, then your
|
||||||
|
license from a particular copyright holder is reinstated (a)
|
||||||
|
provisionally, unless and until the copyright holder explicitly and
|
||||||
|
finally terminates your license, and (b) permanently, if the copyright
|
||||||
|
holder fails to notify you of the violation by some reasonable means
|
||||||
|
prior to 60 days after the cessation.
|
||||||
|
|
||||||
|
Moreover, your license from a particular copyright holder is
|
||||||
|
reinstated permanently if the copyright holder notifies you of the
|
||||||
|
violation by some reasonable means, this is the first time you have
|
||||||
|
received notice of violation of this License (for any work) from that
|
||||||
|
copyright holder, and you cure the violation prior to 30 days after
|
||||||
|
your receipt of the notice.
|
||||||
|
|
||||||
|
Termination of your rights under this section does not terminate the
|
||||||
|
licenses of parties who have received copies or rights from you under
|
||||||
|
this License. If your rights have been terminated and not permanently
|
||||||
|
reinstated, you do not qualify to receive new licenses for the same
|
||||||
|
material under section 10.
|
||||||
|
|
||||||
|
9. Acceptance Not Required for Having Copies.
|
||||||
|
|
||||||
|
You are not required to accept this License in order to receive or
|
||||||
|
run a copy of the Program. Ancillary propagation of a covered work
|
||||||
|
occurring solely as a consequence of using peer-to-peer transmission
|
||||||
|
to receive a copy likewise does not require acceptance. However,
|
||||||
|
nothing other than this License grants you permission to propagate or
|
||||||
|
modify any covered work. These actions infringe copyright if you do
|
||||||
|
not accept this License. Therefore, by modifying or propagating a
|
||||||
|
covered work, you indicate your acceptance of this License to do so.
|
||||||
|
|
||||||
|
10. Automatic Licensing of Downstream Recipients.
|
||||||
|
|
||||||
|
Each time you convey a covered work, the recipient automatically
|
||||||
|
receives a license from the original licensors, to run, modify and
|
||||||
|
propagate that work, subject to this License. You are not responsible
|
||||||
|
for enforcing compliance by third parties with this License.
|
||||||
|
|
||||||
|
An "entity transaction" is a transaction transferring control of an
|
||||||
|
organization, or substantially all assets of one, or subdividing an
|
||||||
|
organization, or merging organizations. If propagation of a covered
|
||||||
|
work results from an entity transaction, each party to that
|
||||||
|
transaction who receives a copy of the work also receives whatever
|
||||||
|
licenses to the work the party's predecessor in interest had or could
|
||||||
|
give under the previous paragraph, plus a right to possession of the
|
||||||
|
Corresponding Source of the work from the predecessor in interest, if
|
||||||
|
the predecessor has it or can get it with reasonable efforts.
|
||||||
|
|
||||||
|
You may not impose any further restrictions on the exercise of the
|
||||||
|
rights granted or affirmed under this License. For example, you may
|
||||||
|
not impose a license fee, royalty, or other charge for exercise of
|
||||||
|
rights granted under this License, and you may not initiate litigation
|
||||||
|
(including a cross-claim or counterclaim in a lawsuit) alleging that
|
||||||
|
any patent claim is infringed by making, using, selling, offering for
|
||||||
|
sale, or importing the Program or any portion of it.
|
||||||
|
|
||||||
|
11. Patents.
|
||||||
|
|
||||||
|
A "contributor" is a copyright holder who authorizes use under this
|
||||||
|
License of the Program or a work on which the Program is based. The
|
||||||
|
work thus licensed is called the contributor's "contributor version".
|
||||||
|
|
||||||
|
A contributor's "essential patent claims" are all patent claims
|
||||||
|
owned or controlled by the contributor, whether already acquired or
|
||||||
|
hereafter acquired, that would be infringed by some manner, permitted
|
||||||
|
by this License, of making, using, or selling its contributor version,
|
||||||
|
but do not include claims that would be infringed only as a
|
||||||
|
consequence of further modification of the contributor version. For
|
||||||
|
purposes of this definition, "control" includes the right to grant
|
||||||
|
patent sublicenses in a manner consistent with the requirements of
|
||||||
|
this License.
|
||||||
|
|
||||||
|
Each contributor grants you a non-exclusive, worldwide, royalty-free
|
||||||
|
patent license under the contributor's essential patent claims, to
|
||||||
|
make, use, sell, offer for sale, import and otherwise run, modify and
|
||||||
|
propagate the contents of its contributor version.
|
||||||
|
|
||||||
|
In the following three paragraphs, a "patent license" is any express
|
||||||
|
agreement or commitment, however denominated, not to enforce a patent
|
||||||
|
(such as an express permission to practice a patent or covenant not to
|
||||||
|
sue for patent infringement). To "grant" such a patent license to a
|
||||||
|
party means to make such an agreement or commitment not to enforce a
|
||||||
|
patent against the party.
|
||||||
|
|
||||||
|
If you convey a covered work, knowingly relying on a patent license,
|
||||||
|
and the Corresponding Source of the work is not available for anyone
|
||||||
|
to copy, free of charge and under the terms of this License, through a
|
||||||
|
publicly available network server or other readily accessible means,
|
||||||
|
then you must either (1) cause the Corresponding Source to be so
|
||||||
|
available, or (2) arrange to deprive yourself of the benefit of the
|
||||||
|
patent license for this particular work, or (3) arrange, in a manner
|
||||||
|
consistent with the requirements of this License, to extend the patent
|
||||||
|
license to downstream recipients. "Knowingly relying" means you have
|
||||||
|
actual knowledge that, but for the patent license, your conveying the
|
||||||
|
covered work in a country, or your recipient's use of the covered work
|
||||||
|
in a country, would infringe one or more identifiable patents in that
|
||||||
|
country that you have reason to believe are valid.
|
||||||
|
|
||||||
|
If, pursuant to or in connection with a single transaction or
|
||||||
|
arrangement, you convey, or propagate by procuring conveyance of, a
|
||||||
|
covered work, and grant a patent license to some of the parties
|
||||||
|
receiving the covered work authorizing them to use, propagate, modify
|
||||||
|
or convey a specific copy of the covered work, then the patent license
|
||||||
|
you grant is automatically extended to all recipients of the covered
|
||||||
|
work and works based on it.
|
||||||
|
|
||||||
|
A patent license is "discriminatory" if it does not include within
|
||||||
|
the scope of its coverage, prohibits the exercise of, or is
|
||||||
|
conditioned on the non-exercise of one or more of the rights that are
|
||||||
|
specifically granted under this License. You may not convey a covered
|
||||||
|
work if you are a party to an arrangement with a third party that is
|
||||||
|
in the business of distributing software, under which you make payment
|
||||||
|
to the third party based on the extent of your activity of conveying
|
||||||
|
the work, and under which the third party grants, to any of the
|
||||||
|
parties who would receive the covered work from you, a discriminatory
|
||||||
|
patent license (a) in connection with copies of the covered work
|
||||||
|
conveyed by you (or copies made from those copies), or (b) primarily
|
||||||
|
for and in connection with specific products or compilations that
|
||||||
|
contain the covered work, unless you entered into that arrangement,
|
||||||
|
or that patent license was granted, prior to 28 March 2007.
|
||||||
|
|
||||||
|
Nothing in this License shall be construed as excluding or limiting
|
||||||
|
any implied license or other defenses to infringement that may
|
||||||
|
otherwise be available to you under applicable patent law.
|
||||||
|
|
||||||
|
12. No Surrender of Others' Freedom.
|
||||||
|
|
||||||
|
If conditions are imposed on you (whether by court order, agreement or
|
||||||
|
otherwise) that contradict the conditions of this License, they do not
|
||||||
|
excuse you from the conditions of this License. If you cannot convey a
|
||||||
|
covered work so as to satisfy simultaneously your obligations under this
|
||||||
|
License and any other pertinent obligations, then as a consequence you may
|
||||||
|
not convey it at all. For example, if you agree to terms that obligate you
|
||||||
|
to collect a royalty for further conveying from those to whom you convey
|
||||||
|
the Program, the only way you could satisfy both those terms and this
|
||||||
|
License would be to refrain entirely from conveying the Program.
|
||||||
|
|
||||||
|
13. Use with the GNU Affero General Public License.
|
||||||
|
|
||||||
|
Notwithstanding any other provision of this License, you have
|
||||||
|
permission to link or combine any covered work with a work licensed
|
||||||
|
under version 3 of the GNU Affero General Public License into a single
|
||||||
|
combined work, and to convey the resulting work. The terms of this
|
||||||
|
License will continue to apply to the part which is the covered work,
|
||||||
|
but the special requirements of the GNU Affero General Public License,
|
||||||
|
section 13, concerning interaction through a network will apply to the
|
||||||
|
combination as such.
|
||||||
|
|
||||||
|
14. Revised Versions of this License.
|
||||||
|
|
||||||
|
The Free Software Foundation may publish revised and/or new versions of
|
||||||
|
the GNU General Public License from time to time. Such new versions will
|
||||||
|
be similar in spirit to the present version, but may differ in detail to
|
||||||
|
address new problems or concerns.
|
||||||
|
|
||||||
|
Each version is given a distinguishing version number. If the
|
||||||
|
Program specifies that a certain numbered version of the GNU General
|
||||||
|
Public License "or any later version" applies to it, you have the
|
||||||
|
option of following the terms and conditions either of that numbered
|
||||||
|
version or of any later version published by the Free Software
|
||||||
|
Foundation. If the Program does not specify a version number of the
|
||||||
|
GNU General Public License, you may choose any version ever published
|
||||||
|
by the Free Software Foundation.
|
||||||
|
|
||||||
|
If the Program specifies that a proxy can decide which future
|
||||||
|
versions of the GNU General Public License can be used, that proxy's
|
||||||
|
public statement of acceptance of a version permanently authorizes you
|
||||||
|
to choose that version for the Program.
|
||||||
|
|
||||||
|
Later license versions may give you additional or different
|
||||||
|
permissions. However, no additional obligations are imposed on any
|
||||||
|
author or copyright holder as a result of your choosing to follow a
|
||||||
|
later version.
|
||||||
|
|
||||||
|
15. Disclaimer of Warranty.
|
||||||
|
|
||||||
|
THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY
|
||||||
|
APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT
|
||||||
|
HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY
|
||||||
|
OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO,
|
||||||
|
THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
||||||
|
PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM
|
||||||
|
IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF
|
||||||
|
ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
|
||||||
|
|
||||||
|
16. Limitation of Liability.
|
||||||
|
|
||||||
|
IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
|
||||||
|
WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS
|
||||||
|
THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY
|
||||||
|
GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE
|
||||||
|
USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF
|
||||||
|
DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD
|
||||||
|
PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),
|
||||||
|
EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF
|
||||||
|
SUCH DAMAGES.
|
||||||
|
|
||||||
|
17. Interpretation of Sections 15 and 16.
|
||||||
|
|
||||||
|
If the disclaimer of warranty and limitation of liability provided
|
||||||
|
above cannot be given local legal effect according to their terms,
|
||||||
|
reviewing courts shall apply local law that most closely approximates
|
||||||
|
an absolute waiver of all civil liability in connection with the
|
||||||
|
Program, unless a warranty or assumption of liability accompanies a
|
||||||
|
copy of the Program in return for a fee.
|
||||||
|
|
||||||
|
END OF TERMS AND CONDITIONS
|
||||||
|
|
||||||
|
How to Apply These Terms to Your New Programs
|
||||||
|
|
||||||
|
If you develop a new program, and you want it to be of the greatest
|
||||||
|
possible use to the public, the best way to achieve this is to make it
|
||||||
|
free software which everyone can redistribute and change under these terms.
|
||||||
|
|
||||||
|
To do so, attach the following notices to the program. It is safest
|
||||||
|
to attach them to the start of each source file to most effectively
|
||||||
|
state the exclusion of warranty; and each file should have at least
|
||||||
|
the "copyright" line and a pointer to where the full notice is found.
|
||||||
|
|
||||||
|
<one line to give the program's name and a brief idea of what it does.>
|
||||||
|
Copyright (C) <year> <name of author>
|
||||||
|
|
||||||
|
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 <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
Also add information on how to contact you by electronic and paper mail.
|
||||||
|
|
||||||
|
If the program does terminal interaction, make it output a short
|
||||||
|
notice like this when it starts in an interactive mode:
|
||||||
|
|
||||||
|
<program> Copyright (C) <year> <name of author>
|
||||||
|
This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
|
||||||
|
This is free software, and you are welcome to redistribute it
|
||||||
|
under certain conditions; type `show c' for details.
|
||||||
|
|
||||||
|
The hypothetical commands `show w' and `show c' should show the appropriate
|
||||||
|
parts of the General Public License. Of course, your program's commands
|
||||||
|
might be different; for a GUI interface, you would use an "about box".
|
||||||
|
|
||||||
|
You should also get your employer (if you work as a programmer) or school,
|
||||||
|
if any, to sign a "copyright disclaimer" for the program, if necessary.
|
||||||
|
For more information on this, and how to apply and follow the GNU GPL, see
|
||||||
|
<https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
The GNU General Public License does not permit incorporating your program
|
||||||
|
into proprietary programs. If your program is a subroutine library, you
|
||||||
|
may consider it more useful to permit linking proprietary applications with
|
||||||
|
the library. If this is what you want to do, use the GNU Lesser General
|
||||||
|
Public License instead of this License. But first, please read
|
||||||
|
<https://www.gnu.org/licenses/why-not-lgpl.html>.
|
||||||
|
75
Makefile
@ -1,42 +1,79 @@
|
|||||||
.PHONY: all nro 3dsx cia clean linux
|
.PHONY: all nro 3dsx cia clean linux 3dslink nxlink format release release-3dsx release-cia release-3ds release-nro
|
||||||
|
|
||||||
|
TARGET := $(notdir $(CURDIR))
|
||||||
|
|
||||||
export GITREV := $(shell git rev-parse HEAD 2>/dev/null | cut -c1-8)
|
export GITREV := $(shell git rev-parse HEAD 2>/dev/null | cut -c1-8)
|
||||||
export VERSION_MAJOR := 2
|
export VERSION_MAJOR := 3
|
||||||
export VERSION_MINOR := 3
|
export VERSION_MINOR := 0
|
||||||
export VERSION_MICRO := 1
|
export VERSION_MICRO := 0
|
||||||
export VERSION := $(VERSION_MAJOR).$(VERSION_MINOR).$(VERSION_MICRO)
|
export VERSION := $(VERSION_MAJOR).$(VERSION_MINOR).$(VERSION_MICRO)-rc1
|
||||||
|
|
||||||
ifneq ($(strip $(GITREV)),)
|
ifneq ($(strip $(GITREV)),)
|
||||||
export VERSION := $(VERSION)-$(GITREV)
|
export VERSION := $(VERSION)-$(GITREV)
|
||||||
endif
|
endif
|
||||||
|
|
||||||
all:
|
all: 3dsx nro linux
|
||||||
@echo please choose 3dsx, cia, linux, or nro
|
|
||||||
|
|
||||||
release:
|
nxlink:
|
||||||
# can't let these three run in parallel with each other due to using same
|
@$(MAKE) -f Makefile.switch nxlink
|
||||||
# ftpd.elf file name
|
|
||||||
@$(MAKE) -f Makefile.switch all
|
3dslink: 3dsx
|
||||||
@$(MAKE) -f Makefile.3ds 3dsx
|
@/opt/devkitpro/tools/bin/3dslink $(TARGET)-3ds.3dsx
|
||||||
@$(MAKE) -f Makefile.3ds cia
|
|
||||||
@xz -c <ftpd.3dsx >ftpd.3dsx.xz
|
format:
|
||||||
@xz -c <ftpd.cia >ftpd.cia.xz
|
@clang-format -style=file -i $(filter-out \
|
||||||
@xz -c <ftpd.nro >ftpd.nro.xz
|
include/imgui.h \
|
||||||
|
source/pc/imgui_impl_glfw.cpp \
|
||||||
|
source/pc/imgui_impl_glfw.h \
|
||||||
|
source/pc/imgui_impl_opengl3.cpp \
|
||||||
|
source/pc/imgui_impl_opengl3.h \
|
||||||
|
source/pc/KHR/khrplatform.h \
|
||||||
|
source/pc/glad.c \
|
||||||
|
source/pc/glad/glad.h \
|
||||||
|
source/imgui/imgui.cpp \
|
||||||
|
source/imgui/imgui_demo.cpp \
|
||||||
|
source/imgui/imgui_draw.cpp \
|
||||||
|
source/imgui/imgui_widgets.cpp \
|
||||||
|
source/imgui/imstb_rectpack.h \
|
||||||
|
source/imgui/imstb_textedit.h \
|
||||||
|
source/imgui/imstb_truetype.h \
|
||||||
|
source/imgui/imgui_internal.h, \
|
||||||
|
$(shell find source include -type f -name \*.c -o -name \*.cpp -o -name \*.h))
|
||||||
|
|
||||||
|
release: release-3ds release-nro
|
||||||
|
@xz -c <$(TARGET)-3ds.3dsx >ftpd.3dsx.xz
|
||||||
|
@echo xz -c <$(TARGET)-3ds.cia >ftpd.cia.xz
|
||||||
|
@echo xz -c <$(TARGET)-nx.nro >ftpd.nro.xz
|
||||||
|
|
||||||
nro:
|
nro:
|
||||||
@$(MAKE) -f Makefile.switch all
|
@$(MAKE) -f Makefile.switch all
|
||||||
|
|
||||||
|
release-nro:
|
||||||
|
@$(MAKE) DEFINES=-DNDEBUG -f Makefile.switch all
|
||||||
|
|
||||||
3dsx:
|
3dsx:
|
||||||
@$(MAKE) -f Makefile.3ds 3dsx
|
@$(MAKE) -f Makefile.3ds 3dsx
|
||||||
|
|
||||||
cia:
|
release-3dsx:
|
||||||
|
@$(MAKE) DEFINES=-DNDEBUG -f Makefile.3ds 3dsx
|
||||||
|
|
||||||
|
cia: 3dsx
|
||||||
@$(MAKE) -f Makefile.3ds cia
|
@$(MAKE) -f Makefile.3ds cia
|
||||||
|
|
||||||
|
release-cia: release-3dsx
|
||||||
|
@$(MAKE) DEFINES=-NDEBUG -f Makefile.3ds cia
|
||||||
|
|
||||||
|
release-3ds:
|
||||||
|
# can't let these three run in parallel with each other due to using same
|
||||||
|
# .elf file name
|
||||||
|
@$(MAKE) DEFINES=-DNDEBUG -f Makefile.3ds 3dsx
|
||||||
|
@$(MAKE) DEFINES=-DNDEBUG -f Makefile.3ds cia
|
||||||
|
|
||||||
linux:
|
linux:
|
||||||
@$(MAKE) -f Makefile.linux
|
@$(MAKE) -f Makefile.linux
|
||||||
|
|
||||||
clean:
|
clean:
|
||||||
@$(MAKE) -f Makefile.switch clean
|
@$(MAKE) -f Makefile.switch clean
|
||||||
@$(MAKE) -f Makefile.3ds clean
|
@$(MAKE) -f Makefile.3ds clean
|
||||||
@$(MAKE) -f Makefile.linux clean
|
@$(MAKE) -f Makefile.linux clean
|
||||||
@$(RM) ftpd.3dsx.xz ftpd.cia.xz ftpd.nro.xz
|
@$(RM) ftpd.3dsx.xz ftpd.cia.xz ftpd.nro.xz
|
||||||
|
182
Makefile.3ds
@ -15,6 +15,10 @@ include $(DEVKITARM)/3ds_rules
|
|||||||
# 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
|
||||||
# INCLUDES is a list of directories containing header files
|
# INCLUDES is a list of directories containing header files
|
||||||
|
# GRAPHICS is a list of directories containing graphics files
|
||||||
|
# GFXBUILD is the directory where converted graphics files will be placed
|
||||||
|
# If set to $(BUILD), it will statically link in the converted
|
||||||
|
# files as if they were data files.
|
||||||
#
|
#
|
||||||
# NO_SMDH: if set to anything, no SMDH file is generated.
|
# NO_SMDH: if set to anything, no SMDH file is generated.
|
||||||
# ROMFS is the directory which contains the RomFS, relative to the Makefile (Optional)
|
# ROMFS is the directory which contains the RomFS, relative to the Makefile (Optional)
|
||||||
@ -27,14 +31,16 @@ include $(DEVKITARM)/3ds_rules
|
|||||||
# - icon.png
|
# - icon.png
|
||||||
# - <libctru folder>/default_icon.png
|
# - <libctru folder>/default_icon.png
|
||||||
#---------------------------------------------------------------------------------
|
#---------------------------------------------------------------------------------
|
||||||
TARGET := ftpd
|
TARGET := $(notdir $(CURDIR))-3ds
|
||||||
BUILD := build.3ds
|
BUILD := build.3ds
|
||||||
SOURCES := source
|
SOURCES := source source/3ds source/imgui
|
||||||
DATA := data
|
DATA := data
|
||||||
INCLUDES := include
|
INCLUDES := include
|
||||||
ROMFS :=
|
GRAPHICS := gfx.3ds
|
||||||
|
ROMFS := romfs.3ds
|
||||||
|
GFXBUILD := $(ROMFS)
|
||||||
|
|
||||||
APP_TITLE := ftpd snap!
|
APP_TITLE := ftpd
|
||||||
APP_DESCRIPTION := v$(VERSION)
|
APP_DESCRIPTION := v$(VERSION)
|
||||||
APP_AUTHOR := mtheall
|
APP_AUTHOR := mtheall
|
||||||
|
|
||||||
@ -46,21 +52,23 @@ RSF_FILE := meta/ftpd-cia.rsf
|
|||||||
#---------------------------------------------------------------------------------
|
#---------------------------------------------------------------------------------
|
||||||
# options for code generation
|
# options for code generation
|
||||||
#---------------------------------------------------------------------------------
|
#---------------------------------------------------------------------------------
|
||||||
|
OPTIMIZE := -O2
|
||||||
ARCH := -march=armv6k -mtune=mpcore -mfloat-abi=hard -mtp=soft
|
ARCH := -march=armv6k -mtune=mpcore -mfloat-abi=hard -mtp=soft
|
||||||
|
|
||||||
CFLAGS := -g -Wall -O3 -mword-relocations \
|
CFLAGS := -g -Wall $(OPTIMIZE) -mword-relocations \
|
||||||
-fomit-frame-pointer -ffunction-sections \
|
-fomit-frame-pointer -ffunction-sections -fdata-sections \
|
||||||
$(ARCH) \
|
$(ARCH) $(DEFINES)
|
||||||
-DSTATUS_STRING="\"ftpd v$(VERSION)\""
|
|
||||||
|
|
||||||
CFLAGS += $(INCLUDE) -DARM11 -D_3DS
|
CFLAGS += $(INCLUDE) -DARM11 -D_3DS \
|
||||||
|
-DSTATUS_STRING="\"ftpd v$(VERSION)\"" \
|
||||||
|
-DIMGUI_DISABLE_INCLUDE_IMCONFIG_H=1
|
||||||
|
|
||||||
CXXFLAGS := $(CFLAGS) -fno-rtti -fno-exceptions -std=gnu++11
|
CXXFLAGS := $(CFLAGS) -fno-rtti -fno-exceptions -std=gnu++17
|
||||||
|
|
||||||
ASFLAGS := -g $(ARCH)
|
ASFLAGS := -g $(ARCH)
|
||||||
LDFLAGS = -specs=3dsx.specs -g $(ARCH) -Wl,-Map,$(TARGET).map
|
LDFLAGS = -specs=3dsx.specs -g $(ARCH) -Wl,-Map,$(TARGET).map $(OPTIMIZE)
|
||||||
|
|
||||||
LIBS := -lctru -lm
|
LIBS := -lcitro3d -lctru -lm
|
||||||
|
|
||||||
#---------------------------------------------------------------------------------
|
#---------------------------------------------------------------------------------
|
||||||
# list of directories containing libraries, this must be the top level containing
|
# list of directories containing libraries, this must be the top level containing
|
||||||
@ -80,14 +88,18 @@ export OUTPUT := $(CURDIR)/$(TARGET)
|
|||||||
export TOPDIR := $(CURDIR)
|
export TOPDIR := $(CURDIR)
|
||||||
|
|
||||||
export VPATH := $(foreach dir,$(SOURCES),$(CURDIR)/$(dir)) \
|
export VPATH := $(foreach dir,$(SOURCES),$(CURDIR)/$(dir)) \
|
||||||
|
$(foreach dir,$(GRAPHICS),$(CURDIR)/$(dir)) \
|
||||||
$(foreach dir,$(DATA),$(CURDIR)/$(dir))
|
$(foreach dir,$(DATA),$(CURDIR)/$(dir))
|
||||||
|
|
||||||
export DEPSDIR := $(CURDIR)/$(BUILD)
|
export DEPSDIR := $(CURDIR)/$(BUILD)
|
||||||
|
|
||||||
CFILES := $(foreach dir,$(SOURCES),$(notdir $(wildcard $(dir)/*.c)))
|
CFILES := $(foreach dir,$(SOURCES),$(notdir $(wildcard $(dir)/*.c)))
|
||||||
CPPFILES := $(foreach dir,$(SOURCES),$(notdir $(wildcard $(dir)/*.cpp)))
|
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)/*.*)))
|
PICAFILES := $(foreach dir,$(SOURCES),$(notdir $(wildcard $(dir)/*.v.pica)))
|
||||||
|
SHLISTFILES := $(foreach dir,$(SOURCES),$(notdir $(wildcard $(dir)/*.shlist)))
|
||||||
|
GFXFILES := $(foreach dir,$(GRAPHICS),$(notdir $(wildcard $(dir)/*.t3s)))
|
||||||
|
BINFILES := $(foreach dir,$(DATA),$(notdir $(wildcard $(dir)/*.*)))
|
||||||
|
|
||||||
#---------------------------------------------------------------------------------
|
#---------------------------------------------------------------------------------
|
||||||
# use CXX for linking C++ projects, CC for standard C
|
# use CXX for linking C++ projects, CC for standard C
|
||||||
@ -99,16 +111,38 @@ else
|
|||||||
endif
|
endif
|
||||||
#---------------------------------------------------------------------------------
|
#---------------------------------------------------------------------------------
|
||||||
|
|
||||||
export OFILES := $(addsuffix .o,$(BINFILES)) \
|
#---------------------------------------------------------------------------------
|
||||||
$(CPPFILES:.cpp=.o) \
|
ifeq ($(GFXBUILD),$(BUILD))
|
||||||
$(CFILES:.c=.o) \
|
#---------------------------------------------------------------------------------
|
||||||
$(SFILES:.s=.o)
|
export T3XFILES := $(GFXFILES:.t3s=.t3x)
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
else
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
export ROMFS_T3XFILES := $(patsubst %.t3s,$(GFXBUILD)/%.t3x,$(GFXFILES))
|
||||||
|
export T3XHFILES := $(patsubst %.t3s,$(BUILD)/%.h,$(GFXFILES))
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
endif
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
|
||||||
export INCLUDE := $(foreach dir,$(INCLUDES),-I$(CURDIR)/$(dir)) \
|
export OFILES_SOURCES := $(CPPFILES:.cpp=.o) $(CFILES:.c=.o) $(SFILES:.s=.o)
|
||||||
$(foreach dir,$(LIBDIRS),-I$(dir)/include) \
|
|
||||||
-I$(CURDIR)/$(BUILD)
|
|
||||||
|
|
||||||
export LIBPATHS := $(foreach dir,$(LIBDIRS),-L$(dir)/lib)
|
export OFILES_BIN := $(addsuffix .o,$(BINFILES)) \
|
||||||
|
$(PICAFILES:.v.pica=.shbin.o) $(SHLISTFILES:.shlist=.shbin.o) \
|
||||||
|
$(addsuffix .o,$(T3XFILES))
|
||||||
|
|
||||||
|
export OFILES := $(OFILES_BIN) $(OFILES_SOURCES)
|
||||||
|
|
||||||
|
export HFILES := $(PICAFILES:.v.pica=_shbin.h) $(SHLISTFILES:.shlist=_shbin.h) \
|
||||||
|
$(addsuffix .h,$(subst .,_,$(BINFILES))) \
|
||||||
|
$(GFXFILES:.t3s=.h)
|
||||||
|
|
||||||
|
export INCLUDE := $(foreach dir,$(INCLUDES),-I$(CURDIR)/$(dir)) \
|
||||||
|
$(foreach dir,$(LIBDIRS),-I$(dir)/include) \
|
||||||
|
-I$(CURDIR)/$(BUILD)
|
||||||
|
|
||||||
|
export LIBPATHS := $(foreach dir,$(LIBDIRS),-L$(dir)/lib)
|
||||||
|
|
||||||
|
export _3DSXDEPS := $(if $(NO_SMDH),,$(OUTPUT).smdh)
|
||||||
|
|
||||||
ifeq ($(strip $(ICON)),)
|
ifeq ($(strip $(ICON)),)
|
||||||
icons := $(wildcard *.png)
|
icons := $(wildcard *.png)
|
||||||
@ -131,36 +165,54 @@ ifneq ($(ROMFS),)
|
|||||||
export _3DSXFLAGS += --romfs=$(CURDIR)/$(ROMFS)
|
export _3DSXFLAGS += --romfs=$(CURDIR)/$(ROMFS)
|
||||||
endif
|
endif
|
||||||
|
|
||||||
.PHONY: $(BUILD) clean all
|
.PHONY: $(BUILD) clean all 3dsx cia
|
||||||
|
|
||||||
#---------------------------------------------------------------------------------
|
#---------------------------------------------------------------------------------
|
||||||
all: $(BUILD)
|
all: $(BUILD) $(GFXBUILD) $(DEPSDIR) $(ROMFS_T3XFILES) $(T3XHFILES)
|
||||||
@$(MAKE) --no-print-directory -C $(BUILD) -f $(CURDIR)/Makefile.3ds
|
@$(MAKE) --no-print-directory -C $(BUILD) -f $(CURDIR)/Makefile.3ds
|
||||||
|
|
||||||
3dsx: $(BUILD)
|
$(BUILD):
|
||||||
|
@mkdir -p $@
|
||||||
|
|
||||||
|
ifneq ($(GFXBUILD),$(BUILD))
|
||||||
|
$(GFXBUILD):
|
||||||
|
@mkdir -p $@
|
||||||
|
endif
|
||||||
|
|
||||||
|
ifneq ($(DEPSDIR),$(BUILD))
|
||||||
|
$(DEPSDIR):
|
||||||
|
mkdir -p $@
|
||||||
|
endif
|
||||||
|
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
$(GFXBUILD)/%.t3x $(BUILD)/%.h: %.t3s
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
@echo $(notdir $<)
|
||||||
|
tex3ds -i $< -H $(BUILD)/$*.h -d $(DEPSDIR)/$*.d -o $(GFXBUILD)/$*.t3x
|
||||||
|
|
||||||
|
3dsx: $(BUILD) $(GFXBUILD) $(DEPSDIR) $(ROMFS_T3XFILES) $(T3XHFILES)
|
||||||
@$(MAKE) --no-print-directory -C $(BUILD) -f $(CURDIR)/Makefile.3ds 3dsx
|
@$(MAKE) --no-print-directory -C $(BUILD) -f $(CURDIR)/Makefile.3ds 3dsx
|
||||||
|
|
||||||
cia: $(BUILD)
|
cia: $(BUILD) $(GFXBUILD) $(DEPSDIR) $(ROMFS_T3XFILES) $(T3XHFILES)
|
||||||
@$(MAKE) --no-print-directory -C $(BUILD) -f $(CURDIR)/Makefile.3ds cia
|
@$(MAKE) --no-print-directory -C $(BUILD) -f $(CURDIR)/Makefile.3ds cia
|
||||||
|
|
||||||
$(BUILD):
|
|
||||||
@[ -d $@ ] || mkdir -p $@
|
|
||||||
|
|
||||||
#---------------------------------------------------------------------------------
|
#---------------------------------------------------------------------------------
|
||||||
clean:
|
clean:
|
||||||
@echo clean ...
|
@echo clean ...
|
||||||
@$(RM) -r $(BUILD) $(TARGET).3dsx $(OUTPUT).smdh $(TARGET).elf $(TARGET).cia output/
|
@$(RM) -r $(BUILD) \
|
||||||
|
$(TARGET).3dsx \
|
||||||
|
$(OUTPUT).smdh \
|
||||||
|
$(TARGET).elf \
|
||||||
|
$(TARGET).cia \
|
||||||
|
$(ROMFS_T3XFILES) \
|
||||||
|
output/
|
||||||
|
|
||||||
#---------------------------------------------------------------------------------
|
#---------------------------------------------------------------------------------
|
||||||
else
|
else
|
||||||
|
|
||||||
DEPENDS := $(OFILES:.o=.d)
|
|
||||||
|
|
||||||
#---------------------------------------------------------------------------------
|
#---------------------------------------------------------------------------------
|
||||||
# main targets
|
# main targets
|
||||||
#---------------------------------------------------------------------------------
|
#---------------------------------------------------------------------------------
|
||||||
#all: $(OUTPUT).cia $(OUTPUT).3dsx
|
all: $(OUTPUT).cia $(OUTPUT).3dsx
|
||||||
|
|
||||||
3dsx: $(OUTPUT).3dsx
|
3dsx: $(OUTPUT).3dsx
|
||||||
|
|
||||||
@ -173,11 +225,56 @@ $(OUTPUT).smdh: $(TOPDIR)/Makefile $(TOPDIR)/Makefile.3ds
|
|||||||
$(OUTPUT).3dsx: $(OUTPUT).smdh
|
$(OUTPUT).3dsx: $(OUTPUT).smdh
|
||||||
endif
|
endif
|
||||||
|
|
||||||
$(OUTPUT).3dsx: $(OUTPUT).elf
|
$(OUTPUT).3dsx: $(OUTPUT).elf $(_3DSXDEPS)
|
||||||
|
|
||||||
|
$(OFILES_SOURCES): $(HFILES)
|
||||||
|
|
||||||
$(OUTPUT).elf: $(OFILES)
|
$(OUTPUT).elf: $(OFILES)
|
||||||
|
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
# you need a rule like this for each extension you use as binary data
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
%.bin.o %_bin.h: %.bin
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
@echo $(notdir $<)
|
||||||
|
@$(bin2o)
|
||||||
|
|
||||||
$(OFILES): $(TOPDIR)/Makefile $(TOPDIR)/Makefile.3ds
|
$(OFILES): $(TOPDIR)/Makefile $(TOPDIR)/Makefile.3ds
|
||||||
|
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
.PRECIOUS: %.t3x
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
%.t3x.o %_t3x.h: %.t3x
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
@echo $(notdir $<)
|
||||||
|
@$(bin2o)
|
||||||
|
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
# rules for assembling GPU shaders
|
||||||
|
#---------------------------------------------------------------------------------
|
||||||
|
define shader-as
|
||||||
|
$(eval CURBIN := $*.shbin)
|
||||||
|
$(eval DEPSFILE := $(DEPSDIR)/$*.shbin.d)
|
||||||
|
echo "$(CURBIN).o: $< $1" > $(DEPSFILE)
|
||||||
|
echo "extern const u8" `(echo $(CURBIN) | sed -e 's/^\([0-9]\)/_\1/' | tr . _)`"_end[];" > `(echo $(CURBIN) | tr . _)`.h
|
||||||
|
echo "extern const u8" `(echo $(CURBIN) | sed -e 's/^\([0-9]\)/_\1/' | tr . _)`"[];" >> `(echo $(CURBIN) | tr . _)`.h
|
||||||
|
echo "extern const u32" `(echo $(CURBIN) | sed -e 's/^\([0-9]\)/_\1/' | tr . _)`_size";" >> `(echo $(CURBIN) | tr . _)`.h
|
||||||
|
picasso -o $(CURBIN) $1
|
||||||
|
bin2s $(CURBIN) | $(AS) -o $*.shbin.o
|
||||||
|
endef
|
||||||
|
|
||||||
|
%.shbin.o %_shbin.h : %.v.pica %.g.pica
|
||||||
|
@echo $(notdir $^)
|
||||||
|
@$(call shader-as,$^)
|
||||||
|
|
||||||
|
%.shbin.o %_shbin.h : %.v.pica
|
||||||
|
@echo $(notdir $<)
|
||||||
|
@$(call shader-as,$<)
|
||||||
|
|
||||||
|
%.shbin.o %_shbin.h : %.shlist
|
||||||
|
@echo $(notdir $<)
|
||||||
|
@$(call shader-as,$(foreach file,$(shell cat $<),$(dir $<)$(file)))
|
||||||
|
|
||||||
$(OUTPUT).cia: $(OUTPUT).elf $(OUTPUT).smdh $(TARGET).bnr $(TOPDIR)/$(RSF_FILE)
|
$(OUTPUT).cia: $(OUTPUT).elf $(OUTPUT).smdh $(TARGET).bnr $(TOPDIR)/$(RSF_FILE)
|
||||||
@makerom -f cia -target t -exefslogo -o $@ \
|
@makerom -f cia -target t -exefslogo -o $@ \
|
||||||
-elf $(OUTPUT).elf -rsf $(TOPDIR)/$(RSF_FILE) \
|
-elf $(OUTPUT).elf -rsf $(TOPDIR)/$(RSF_FILE) \
|
||||||
@ -190,15 +287,8 @@ $(TARGET).bnr: $(TOPDIR)/$(BNR_IMAGE) $(TOPDIR)/$(BNR_AUDIO)
|
|||||||
@bannertool makebanner -o $@ -i $(TOPDIR)/$(BNR_IMAGE) -a $(TOPDIR)/$(BNR_AUDIO)
|
@bannertool makebanner -o $@ -i $(TOPDIR)/$(BNR_IMAGE) -a $(TOPDIR)/$(BNR_AUDIO)
|
||||||
@echo "built ... $@"
|
@echo "built ... $@"
|
||||||
|
|
||||||
#---------------------------------------------------------------------------------
|
|
||||||
# you need a rule like this for each extension you use as binary data
|
|
||||||
#---------------------------------------------------------------------------------
|
|
||||||
%.bin.o: %.bin
|
|
||||||
#---------------------------------------------------------------------------------
|
|
||||||
@echo $(notdir $<)
|
|
||||||
@$(bin2o)
|
|
||||||
|
|
||||||
-include $(DEPENDS)
|
-include $(DEPSDIR)/*.d
|
||||||
|
|
||||||
#---------------------------------------------------------------------------------------
|
#---------------------------------------------------------------------------------------
|
||||||
endif
|
endif
|
||||||
|
@ -1,23 +1,36 @@
|
|||||||
TARGET := ftpd
|
TARGET := ftpd
|
||||||
|
BUILD := build.linux
|
||||||
|
|
||||||
CFILES := $(wildcard source/*.c)
|
CFILES := $(wildcard source/pc/*.c)
|
||||||
OFILES := $(patsubst source/%,build.linux/%,$(CFILES:.c=.o))
|
OFILES := $(patsubst source/%,$(BUILD)/%,$(CFILES:.c=.c.o))
|
||||||
|
CXXFILES := $(wildcard source/*.cpp source/imgui/*.cpp source/pc/*.cpp)
|
||||||
|
OXXFILES := $(patsubst source/%,$(BUILD)/%,$(CXXFILES:.cpp=.cpp.o))
|
||||||
|
|
||||||
CFLAGS := -g -Wall -Iinclude -DSTATUS_STRING="\"ftpd v$(VERSION)\""
|
CPPFLAGS := -g -Wall -pthread -Iinclude -Isource/pc \
|
||||||
LDFLAGS :=
|
`pkg-config --cflags gl glfw3` \
|
||||||
|
-DSTATUS_STRING="\"ftpd v$(VERSION)\"" \
|
||||||
|
-DIMGUI_DISABLE_INCLUDE_IMCONFIG_H=1 \
|
||||||
|
-DIMGUI_IMPL_OPENGL_LOADER_GLAD=1
|
||||||
|
CFLAGS := $(CPPFLAGS)
|
||||||
|
CXXFLAGS := $(CPPFLAGS) -std=gnu++17
|
||||||
|
LDFLAGS := -pthread `pkg-config --libs gl glfw3` -ldl
|
||||||
|
|
||||||
.PHONY: all clean
|
.PHONY: all clean
|
||||||
|
|
||||||
all: build.linux $(TARGET)
|
all: $(TARGET)
|
||||||
|
|
||||||
build.linux:
|
$(TARGET): $(OFILES) $(OXXFILES)
|
||||||
@mkdir build.linux/
|
$(CXX) -o $@ $^ $(LDFLAGS)
|
||||||
|
|
||||||
$(TARGET): $(OFILES)
|
$(OFILES): $(BUILD)/%.c.o : source/%.c
|
||||||
@$(CC) -o $@ $^ $(LDFLAGS)
|
@[ -d $(dir $@) ] || mkdir -p $(dir $@)
|
||||||
|
$(CC) -MMD -MP -MF $(BUILD)/$*.c.d $(CFLAGS) -c $< -o $@
|
||||||
|
|
||||||
$(OFILES): build.linux/%.o : source/%.c
|
$(OXXFILES): $(BUILD)/%.cpp.o : source/%.cpp
|
||||||
@$(CC) -o $@ -c $< $(CFLAGS)
|
@[ -d $(dir $@) ] || mkdir -p $(dir $@)
|
||||||
|
$(CXX) -MMD -MP -MF $(BUILD)/$*.c.d $(CXXFLAGS) -c $< -o $@
|
||||||
|
|
||||||
clean:
|
clean:
|
||||||
@$(RM) -r build.linux/ $(TARGET)
|
@$(RM) -r $(BUILD) $(TARGET)
|
||||||
|
|
||||||
|
-include $(shell find $(BUILD) -name \*.d 2>/dev/null)
|
||||||
|
169
Makefile.switch
@ -15,7 +15,7 @@ include $(DEVKITPRO)/libnx/switch_rules
|
|||||||
# 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
|
||||||
# INCLUDES is a list of directories containing header files
|
# INCLUDES is a list of directories containing header files
|
||||||
# EXEFS_SRC is the optional input directory containing data copied into exefs, if anything this normally should only contain "main.npdm".
|
# ROMFS is the directory containing data to be added to RomFS, relative to the Makefile (Optional)
|
||||||
#
|
#
|
||||||
# NO_ICON: if set to anything, do not use icon.
|
# NO_ICON: if set to anything, do not use icon.
|
||||||
# NO_NACP: if set to anything, no .nacp file is generated.
|
# NO_NACP: if set to anything, no .nacp file is generated.
|
||||||
@ -28,36 +28,51 @@ include $(DEVKITPRO)/libnx/switch_rules
|
|||||||
# - <Project name>.jpg
|
# - <Project name>.jpg
|
||||||
# - icon.jpg
|
# - icon.jpg
|
||||||
# - <libnx folder>/default_icon.jpg
|
# - <libnx folder>/default_icon.jpg
|
||||||
|
#
|
||||||
|
# CONFIG_JSON is the filename of the NPDM config file (.json), relative to the project folder.
|
||||||
|
# If not set, it attempts to use one of the following (in this order):
|
||||||
|
# - <Project name>.json
|
||||||
|
# - config.json
|
||||||
|
# If a JSON file is provided or autodetected, an ExeFS PFS0 (.nsp) is built instead
|
||||||
|
# of a homebrew executable (.nro). This is intended to be used for sysmodules.
|
||||||
|
# NACP building is skipped as well.
|
||||||
#---------------------------------------------------------------------------------
|
#---------------------------------------------------------------------------------
|
||||||
APP_TITLE := ftpd snap! $(VERSION)
|
APP_TITLE := ftpd snap! $(VERSION)
|
||||||
APP_AUTHOR := mtheall, TuxSH, WinterMute
|
APP_AUTHOR := mtheall, TuxSH, WinterMute
|
||||||
ICON := meta/ftpd.jpg
|
ICON := meta/ftpd.jpg
|
||||||
APP_VERSION := $(VERSION)
|
APP_VERSION := $(VERSION)
|
||||||
#---------------------------------------------------------------------------------
|
|
||||||
TARGET := ftpd
|
TARGET := $(notdir $(CURDIR))-nx
|
||||||
BUILD := build.switch
|
BUILD := build.switch
|
||||||
SOURCES := source
|
SOURCES := source source/imgui source/nx
|
||||||
DATA := data
|
DATA := data
|
||||||
INCLUDES := include
|
INCLUDES := include
|
||||||
EXEFS_SRC := exefs_src
|
GRAPHICS := gfx.switch
|
||||||
|
ROMFS := romfs.switch
|
||||||
|
|
||||||
|
# Output folders for autogenerated files in romfs
|
||||||
|
OUT_SHADERS := shaders
|
||||||
|
|
||||||
#---------------------------------------------------------------------------------
|
#---------------------------------------------------------------------------------
|
||||||
# options for code generation
|
# options for code generation
|
||||||
#---------------------------------------------------------------------------------
|
#---------------------------------------------------------------------------------
|
||||||
ARCH := -march=armv8-a -mtp=soft -fPIE
|
ARCH := -march=armv8-a+crc+crypto -mtune=cortex-a57 -mtp=soft -fPIE
|
||||||
|
|
||||||
CFLAGS := -g -Wall -O2 \
|
CFLAGS := -g -Wall -Wno-narrowing -Os -ffunction-sections -fdata-sections -save-temps \
|
||||||
-ffast-math \
|
|
||||||
$(ARCH) $(DEFINES)
|
$(ARCH) $(DEFINES)
|
||||||
|
|
||||||
CFLAGS += $(INCLUDE) -D__SWITCH__ -DSTATUS_STRING="\"ftpd v$(VERSION)\""
|
CFLAGS += $(INCLUDE) -D__SWITCH__ \
|
||||||
|
-DSTATUS_STRING="\"ftpd v$(VERSION)\"" \
|
||||||
|
-DIMGUI_DISABLE_INCLUDE_IMCONFIG_H=1 \
|
||||||
|
`$(PREFIX)pkg-config --cflags libzstd`
|
||||||
|
|
||||||
CXXFLAGS := $(CFLAGS) -fno-rtti -fno-exceptions -std=gnu++11
|
CXXFLAGS := $(CFLAGS) -std=gnu++17 -fno-exceptions -fno-rtti
|
||||||
|
|
||||||
ASFLAGS := -g $(ARCH)
|
ASFLAGS := -g $(ARCH)
|
||||||
LDFLAGS = -specs=$(DEVKITPRO)/libnx/switch.specs -g $(ARCH) -Wl,-Map,$(notdir $*.map)
|
LDFLAGS = -specs=$(DEVKITPRO)/libnx/switch.specs -g $(ARCH) \
|
||||||
|
-Wl,-Map,$(notdir $*.map) -Wl,--gc-sections
|
||||||
|
|
||||||
LIBS := -lnx
|
LIBS := `$(PREFIX)pkg-config --libs libzstd` -ldeko3dd -lnx
|
||||||
|
|
||||||
#---------------------------------------------------------------------------------
|
#---------------------------------------------------------------------------------
|
||||||
# list of directories containing libraries, this must be the top level containing
|
# list of directories containing libraries, this must be the top level containing
|
||||||
@ -84,6 +99,8 @@ export DEPSDIR := $(CURDIR)/$(BUILD)
|
|||||||
CFILES := $(foreach dir,$(SOURCES),$(notdir $(wildcard $(dir)/*.c)))
|
CFILES := $(foreach dir,$(SOURCES),$(notdir $(wildcard $(dir)/*.c)))
|
||||||
CPPFILES := $(foreach dir,$(SOURCES),$(notdir $(wildcard $(dir)/*.cpp)))
|
CPPFILES := $(foreach dir,$(SOURCES),$(notdir $(wildcard $(dir)/*.cpp)))
|
||||||
SFILES := $(foreach dir,$(SOURCES),$(notdir $(wildcard $(dir)/*.s)))
|
SFILES := $(foreach dir,$(SOURCES),$(notdir $(wildcard $(dir)/*.s)))
|
||||||
|
GLSLFILES := $(foreach dir,$(SOURCES),$(notdir $(wildcard $(dir)/*.glsl)))
|
||||||
|
GFXFILES := $(foreach dir,$(GRAPHICS),$(notdir $(wildcard $(dir)/*.png)))
|
||||||
BINFILES := $(foreach dir,$(DATA),$(notdir $(wildcard $(dir)/*.*)))
|
BINFILES := $(foreach dir,$(DATA),$(notdir $(wildcard $(dir)/*.*)))
|
||||||
|
|
||||||
#---------------------------------------------------------------------------------
|
#---------------------------------------------------------------------------------
|
||||||
@ -91,17 +108,35 @@ BINFILES := $(foreach dir,$(DATA),$(notdir $(wildcard $(dir)/*.*)))
|
|||||||
#---------------------------------------------------------------------------------
|
#---------------------------------------------------------------------------------
|
||||||
ifeq ($(strip $(CPPFILES)),)
|
ifeq ($(strip $(CPPFILES)),)
|
||||||
#---------------------------------------------------------------------------------
|
#---------------------------------------------------------------------------------
|
||||||
export LD := $(CC)
|
export LD := $(CC)
|
||||||
#---------------------------------------------------------------------------------
|
#---------------------------------------------------------------------------------
|
||||||
else
|
else
|
||||||
#---------------------------------------------------------------------------------
|
#---------------------------------------------------------------------------------
|
||||||
export LD := $(CXX)
|
export LD := $(CXX)
|
||||||
#---------------------------------------------------------------------------------
|
#---------------------------------------------------------------------------------
|
||||||
endif
|
endif
|
||||||
#---------------------------------------------------------------------------------
|
#---------------------------------------------------------------------------------
|
||||||
|
|
||||||
export OFILES := $(addsuffix .o,$(BINFILES)) \
|
export OFILES_BIN := $(addsuffix .o,$(BINFILES))
|
||||||
$(CPPFILES:.cpp=.o) $(CFILES:.c=.o) $(SFILES:.s=.o)
|
export OFILES_SRC := $(CPPFILES:.cpp=.o) $(CFILES:.c=.o) $(SFILES:.s=.o)
|
||||||
|
export OFILES := $(OFILES_BIN) $(OFILES_SRC)
|
||||||
|
export HFILES_BIN := $(addsuffix .h,$(subst .,_,$(BINFILES)))
|
||||||
|
|
||||||
|
ifneq ($(strip $(ROMFS)),)
|
||||||
|
ROMFS_TARGETS :=
|
||||||
|
ROMFS_FOLDERS :=
|
||||||
|
ifneq ($(strip $(OUT_SHADERS)),)
|
||||||
|
ROMFS_SHADERS := $(ROMFS)/$(OUT_SHADERS)
|
||||||
|
ROMFS_TARGETS += $(patsubst %.glsl, $(ROMFS_SHADERS)/%.dksh, $(GLSLFILES))
|
||||||
|
ROMFS_FOLDERS += $(ROMFS_SHADERS)
|
||||||
|
endif
|
||||||
|
|
||||||
|
ROMFS_GFX := $(addprefix $(ROMFS)/,$(GFXFILES:.png=.rgba.zst))
|
||||||
|
ROMFS_TARGETS += $(ROMFS_GFX)
|
||||||
|
ROMFS_FOLDERS += $(ROMFS)
|
||||||
|
|
||||||
|
export ROMFS_DEPS := $(foreach file,$(ROMFS_TARGETS),$(CURDIR)/$(file))
|
||||||
|
endif
|
||||||
|
|
||||||
export INCLUDE := $(foreach dir,$(INCLUDES),-I$(CURDIR)/$(dir)) \
|
export INCLUDE := $(foreach dir,$(INCLUDES),-I$(CURDIR)/$(dir)) \
|
||||||
$(foreach dir,$(LIBDIRS),-I$(dir)/include) \
|
$(foreach dir,$(LIBDIRS),-I$(dir)/include) \
|
||||||
@ -109,7 +144,18 @@ export INCLUDE := $(foreach dir,$(INCLUDES),-I$(CURDIR)/$(dir)) \
|
|||||||
|
|
||||||
export LIBPATHS := $(foreach dir,$(LIBDIRS),-L$(dir)/lib)
|
export LIBPATHS := $(foreach dir,$(LIBDIRS),-L$(dir)/lib)
|
||||||
|
|
||||||
export BUILD_EXEFS_SRC := $(TOPDIR)/$(EXEFS_SRC)
|
ifeq ($(strip $(CONFIG_JSON)),)
|
||||||
|
jsons := $(wildcard *.json)
|
||||||
|
ifneq (,$(findstring $(TARGET).json,$(jsons)))
|
||||||
|
export APP_JSON := $(TOPDIR)/$(TARGET).json
|
||||||
|
else
|
||||||
|
ifneq (,$(findstring config.json,$(jsons)))
|
||||||
|
export APP_JSON := $(TOPDIR)/config.json
|
||||||
|
endif
|
||||||
|
endif
|
||||||
|
else
|
||||||
|
export APP_JSON := $(TOPDIR)/$(CONFIG_JSON)
|
||||||
|
endif
|
||||||
|
|
||||||
ifeq ($(strip $(ICON)),)
|
ifeq ($(strip $(ICON)),)
|
||||||
icons := $(wildcard *.jpg)
|
icons := $(wildcard *.jpg)
|
||||||
@ -136,19 +182,69 @@ ifneq ($(APP_TITLEID),)
|
|||||||
export NACPFLAGS += --titleid=$(APP_TITLEID)
|
export NACPFLAGS += --titleid=$(APP_TITLEID)
|
||||||
endif
|
endif
|
||||||
|
|
||||||
.PHONY: $(BUILD) clean all
|
ifneq ($(ROMFS),)
|
||||||
|
export NROFLAGS += --romfsdir=$(CURDIR)/$(ROMFS)
|
||||||
|
endif
|
||||||
|
|
||||||
|
.PHONY: all clean
|
||||||
|
|
||||||
#---------------------------------------------------------------------------------
|
#---------------------------------------------------------------------------------
|
||||||
all: $(BUILD)
|
all: $(ROMFS_TARGETS) | $(BUILD)
|
||||||
|
@$(MAKE) --no-print-directory -C $(BUILD) -f $(CURDIR)/Makefile.switch
|
||||||
|
|
||||||
|
nxlink: all
|
||||||
|
@$(MAKE) --no-print-directory -C $(BUILD) -f $(CURDIR)/Makefile.switch nxlink
|
||||||
|
|
||||||
$(BUILD):
|
$(BUILD):
|
||||||
@[ -d $@ ] || mkdir -p $@
|
@mkdir -p $@
|
||||||
@$(MAKE) --no-print-directory -C $(BUILD) -f $(CURDIR)/Makefile.switch
|
|
||||||
|
ifneq ($(strip $(ROMFS_TARGETS)),)
|
||||||
|
|
||||||
|
$(ROMFS_TARGETS): | $(ROMFS_FOLDERS)
|
||||||
|
|
||||||
|
$(ROMFS_FOLDERS):
|
||||||
|
@mkdir -p $@
|
||||||
|
|
||||||
|
$(BUILD)/%.rgba: $(GRAPHICS)/%.png | $(BUILD)
|
||||||
|
@convert $< $@
|
||||||
|
|
||||||
|
$(ROMFS_GFX): $(ROMFS)/%.rgba.zst: $(BUILD)/%.rgba
|
||||||
|
@zstd $< -o $@ --ultra -22
|
||||||
|
|
||||||
|
$(ROMFS_SHADERS)/%_vsh.dksh: %_vsh.glsl
|
||||||
|
@echo {vert} $(notdir $<)
|
||||||
|
@uam -s vert -o $@ $<
|
||||||
|
|
||||||
|
$(ROMFS_SHADERS)/%_tcsh.dksh: %_tcsh.glsl
|
||||||
|
@echo {tess_ctrl} $(notdir $<)
|
||||||
|
@uam -s tess_ctrl -o $@ $<
|
||||||
|
|
||||||
|
$(ROMFS_SHADERS)/%_tesh.dksh: %_tesh.glsl
|
||||||
|
@echo {tess_eval} $(notdir $<)
|
||||||
|
@uam -s tess_eval -o $@ $<
|
||||||
|
|
||||||
|
$(ROMFS_SHADERS)/%_gsh.dksh: %_gsh.glsl
|
||||||
|
@echo {geom} $(notdir $<)
|
||||||
|
@uam -s geom -o $@ $<
|
||||||
|
|
||||||
|
$(ROMFS_SHADERS)/%_fsh.dksh: %_fsh.glsl
|
||||||
|
@echo {frag} $(notdir $<)
|
||||||
|
@uam -s frag -o $@ $<
|
||||||
|
|
||||||
|
$(ROMFS_SHADERS)/%.dksh: %.glsl
|
||||||
|
@echo {comp} $(notdir $<)
|
||||||
|
@uam -s comp -o $@ $<
|
||||||
|
|
||||||
|
endif
|
||||||
|
|
||||||
#---------------------------------------------------------------------------------
|
#---------------------------------------------------------------------------------
|
||||||
clean:
|
clean:
|
||||||
@echo clean ...
|
@echo clean ...
|
||||||
@rm -fr $(BUILD) $(TARGET).pfs0 $(TARGET).nso $(TARGET).nro $(TARGET).nacp $(TARGET).elf
|
ifeq ($(strip $(APP_JSON)),)
|
||||||
|
@$(RM) -r $(BUILD) $(ROMFS)/*.zst $(ROMFS_FOLDERS) $(TARGET).nro $(TARGET).nacp $(TARGET).elf
|
||||||
|
else
|
||||||
|
@$(RM) -r $(BUILD) $(ROMFS_FOLDERS) $(TARGET).nsp $(TARGET).nso $(TARGET).npdm $(TARGET).elf
|
||||||
|
endif
|
||||||
|
|
||||||
|
|
||||||
#---------------------------------------------------------------------------------
|
#---------------------------------------------------------------------------------
|
||||||
@ -160,24 +256,37 @@ DEPENDS := $(OFILES:.o=.d)
|
|||||||
#---------------------------------------------------------------------------------
|
#---------------------------------------------------------------------------------
|
||||||
# main targets
|
# main targets
|
||||||
#---------------------------------------------------------------------------------
|
#---------------------------------------------------------------------------------
|
||||||
all : $(OUTPUT).pfs0 $(OUTPUT).nro
|
ifeq ($(strip $(APP_JSON)),)
|
||||||
|
|
||||||
$(OUTPUT).pfs0 : $(OUTPUT).nso
|
all : $(OUTPUT).nro
|
||||||
|
|
||||||
|
nxlink: $(OUTPUT).nro
|
||||||
|
@nxlink -s $(OUTPUT).nro
|
||||||
|
|
||||||
|
ifeq ($(strip $(NO_NACP)),)
|
||||||
|
$(OUTPUT).nro : $(OUTPUT).elf $(OUTPUT).nacp $(ROMFS_DEPS)
|
||||||
|
else
|
||||||
|
$(OUTPUT).nro : $(OUTPUT).elf $(ROMFS_DEPS)
|
||||||
|
endif
|
||||||
|
|
||||||
|
else
|
||||||
|
|
||||||
|
all : $(OUTPUT).nsp
|
||||||
|
|
||||||
|
$(OUTPUT).nsp : $(OUTPUT).nso $(OUTPUT).npdm
|
||||||
|
|
||||||
$(OUTPUT).nso : $(OUTPUT).elf
|
$(OUTPUT).nso : $(OUTPUT).elf
|
||||||
|
|
||||||
ifeq ($(strip $(NO_NACP)),)
|
|
||||||
$(OUTPUT).nro : $(OUTPUT).elf $(OUTPUT).nacp
|
|
||||||
else
|
|
||||||
$(OUTPUT).nro : $(OUTPUT).elf
|
|
||||||
endif
|
endif
|
||||||
|
|
||||||
$(OUTPUT).elf : $(OFILES)
|
$(OUTPUT).elf : $(OFILES)
|
||||||
|
|
||||||
|
$(OFILES_SRC) : $(HFILES_BIN)
|
||||||
|
|
||||||
#---------------------------------------------------------------------------------
|
#---------------------------------------------------------------------------------
|
||||||
# you need a rule like this for each extension you use as binary data
|
# you need a rule like this for each extension you use as binary data
|
||||||
#---------------------------------------------------------------------------------
|
#---------------------------------------------------------------------------------
|
||||||
%.bin.o : %.bin
|
%.bin.o %_bin.h : %.bin
|
||||||
#---------------------------------------------------------------------------------
|
#---------------------------------------------------------------------------------
|
||||||
@echo $(notdir $<)
|
@echo $(notdir $<)
|
||||||
@$(bin2o)
|
@$(bin2o)
|
||||||
|
16
README.md
@ -10,15 +10,15 @@ FTP Server for 3DS/Switch/Linux.
|
|||||||
|
|
||||||
## Latest Builds
|
## Latest Builds
|
||||||
|
|
||||||
CIA: https://mtheall.com/~mtheall/ftpd.cia
|
CIA: https://mtheall.com/~mtheall/ftpd-3ds.cia
|
||||||
|
|
||||||
3DSX: https://mtheall.com/~mtheall/ftpd.3dsx
|
3DSX: https://mtheall.com/~mtheall/ftpd-3ds.3dsx
|
||||||
|
|
||||||
NRO: https://mtheall.com/~mtheall/ftpd.nro
|
NRO: https://mtheall.com/~mtheall/ftpd-nx.nro
|
||||||
|
|
||||||
CIA QR Code
|
CIA QR Code
|
||||||
|
|
||||||
![ftpd.cia](https://github.com/mtheall/ftpd/raw/master/ftpd_qr.png)
|
![ftpd-3ds.cia](https://github.com/mtheall/ftpd/raw/master/ftpd_qr.png)
|
||||||
|
|
||||||
## Build and install
|
## Build and install
|
||||||
|
|
||||||
@ -26,7 +26,7 @@ You must set up the [development environment](https://devkitpro.org/wiki/Getting
|
|||||||
|
|
||||||
### 3DSX
|
### 3DSX
|
||||||
|
|
||||||
The following pacman packages are required to build `ftpd.3dsx`:
|
The following pacman packages are required to build `ftpd-3ds.3dsx`:
|
||||||
|
|
||||||
3dstools
|
3dstools
|
||||||
devkitARM
|
devkitARM
|
||||||
@ -34,13 +34,13 @@ The following pacman packages are required to build `ftpd.3dsx`:
|
|||||||
|
|
||||||
They are available as part of the `3ds-dev` meta-package.
|
They are available as part of the `3ds-dev` meta-package.
|
||||||
|
|
||||||
Build `ftpd.3dsx`:
|
Build `ftpd-3ds.3dsx`:
|
||||||
|
|
||||||
make 3dsx
|
make 3dsx
|
||||||
|
|
||||||
### NRO
|
### NRO
|
||||||
|
|
||||||
The following pacman packages are required to build `ftpd.nro`:
|
The following pacman packages are required to build `ftpd-nx.nro`:
|
||||||
|
|
||||||
devkitA64
|
devkitA64
|
||||||
libnx
|
libnx
|
||||||
@ -48,7 +48,7 @@ The following pacman packages are required to build `ftpd.nro`:
|
|||||||
|
|
||||||
They are available as part of the `switch-dev` meta-package.
|
They are available as part of the `switch-dev` meta-package.
|
||||||
|
|
||||||
Build `ftpd.nro`:
|
Build `ftpd-nx.nro`:
|
||||||
|
|
||||||
make nro
|
make nro
|
||||||
|
|
||||||
|
9
delog.py
@ -1,9 +0,0 @@
|
|||||||
#!/usr/bin/env python
|
|
||||||
|
|
||||||
import sys
|
|
||||||
import re
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
|
||||||
regex = re.compile('\x1b\[[0-9]*;[0-9]*H')
|
|
||||||
for line in sys.stdin:
|
|
||||||
print(regex.sub('', line).strip())
|
|
BIN
ftpd_qr.png
Before Width: | Height: | Size: 621 B After Width: | Height: | Size: 3.4 KiB |
BIN
gfx.3ds/battery0.png
Normal file
After Width: | Height: | Size: 22 KiB |
BIN
gfx.3ds/battery1.png
Normal file
After Width: | Height: | Size: 22 KiB |
BIN
gfx.3ds/battery2.png
Normal file
After Width: | Height: | Size: 22 KiB |
BIN
gfx.3ds/battery3.png
Normal file
After Width: | Height: | Size: 22 KiB |
BIN
gfx.3ds/battery4.png
Normal file
After Width: | Height: | Size: 22 KiB |
BIN
gfx.3ds/batteryCharge.png
Normal file
After Width: | Height: | Size: 22 KiB |
BIN
gfx.3ds/c3dlogo.png
Normal file
After Width: | Height: | Size: 791 B |
12
gfx.3ds/gfx.t3s
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
-f rgba -z auto --atlas
|
||||||
|
battery0.png
|
||||||
|
battery1.png
|
||||||
|
battery2.png
|
||||||
|
battery3.png
|
||||||
|
battery4.png
|
||||||
|
batteryCharge.png
|
||||||
|
c3dlogo.png
|
||||||
|
wifiNull.png
|
||||||
|
wifi1.png
|
||||||
|
wifi2.png
|
||||||
|
wifi3.png
|
BIN
gfx.3ds/wifi1.png
Normal file
After Width: | Height: | Size: 22 KiB |
BIN
gfx.3ds/wifi2.png
Normal file
After Width: | Height: | Size: 22 KiB |
BIN
gfx.3ds/wifi3.png
Normal file
After Width: | Height: | Size: 22 KiB |
BIN
gfx.3ds/wifiNull.png
Normal file
After Width: | Height: | Size: 22 KiB |
BIN
gfx.switch/deko3d.png
Normal file
After Width: | Height: | Size: 54 KiB |
@ -1,44 +0,0 @@
|
|||||||
#pragma once
|
|
||||||
|
|
||||||
#ifdef _3DS
|
|
||||||
#include <3ds.h>
|
|
||||||
#elif defined(__SWITCH__)
|
|
||||||
#include <switch.h>
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if defined(_3DS) || defined(__SWITCH__)
|
|
||||||
#define ESC(x) "\x1b[" #x
|
|
||||||
#define RESET ESC(0m)
|
|
||||||
#define BLACK ESC(30m)
|
|
||||||
#define RED ESC(31;1m)
|
|
||||||
#define GREEN ESC(32;1m)
|
|
||||||
#define YELLOW ESC(33;1m)
|
|
||||||
#define BLUE ESC(34;1m)
|
|
||||||
#define MAGENTA ESC(35;1m)
|
|
||||||
#define CYAN ESC(36;1m)
|
|
||||||
#define WHITE ESC(37;1m)
|
|
||||||
#else
|
|
||||||
#define ESC(x)
|
|
||||||
#define RESET
|
|
||||||
#define BLACK
|
|
||||||
#define RED
|
|
||||||
#define GREEN
|
|
||||||
#define YELLOW
|
|
||||||
#define BLUE
|
|
||||||
#define MAGENTA
|
|
||||||
#define CYAN
|
|
||||||
#define WHITE
|
|
||||||
#endif
|
|
||||||
|
|
||||||
void console_init(void);
|
|
||||||
|
|
||||||
__attribute__((format(printf,1,2)))
|
|
||||||
void console_set_status(const char *fmt, ...);
|
|
||||||
|
|
||||||
__attribute__((format(printf,1,2)))
|
|
||||||
void console_print(const char *fmt, ...);
|
|
||||||
|
|
||||||
__attribute__((format(printf,1,2)))
|
|
||||||
void debug_print(const char *fmt, ...);
|
|
||||||
|
|
||||||
void console_render(void);
|
|
113
include/fs.h
Normal file
@ -0,0 +1,113 @@
|
|||||||
|
// ftpd is a server implementation based on the following:
|
||||||
|
// - RFC 959 (https://tools.ietf.org/html/rfc959)
|
||||||
|
// - RFC 3659 (https://tools.ietf.org/html/rfc3659)
|
||||||
|
// - suggested implementation details from https://cr.yp.to/ftp/filesystem.html
|
||||||
|
//
|
||||||
|
// Copyright (C) 2020 Michael Theall
|
||||||
|
//
|
||||||
|
// 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 <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <dirent.h>
|
||||||
|
|
||||||
|
#include <cstdint>
|
||||||
|
#include <cstdio>
|
||||||
|
#include <cstdlib>
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
|
namespace fs
|
||||||
|
{
|
||||||
|
std::string printSize (std::uint64_t size_);
|
||||||
|
|
||||||
|
class File
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
~File ();
|
||||||
|
|
||||||
|
File ();
|
||||||
|
|
||||||
|
File (File const &that_) = delete;
|
||||||
|
|
||||||
|
File (File &&that_);
|
||||||
|
|
||||||
|
File &operator= (File const &that_) = delete;
|
||||||
|
|
||||||
|
File &operator= (File &&that_);
|
||||||
|
|
||||||
|
operator bool () const;
|
||||||
|
operator FILE * () const;
|
||||||
|
|
||||||
|
void setBufferSize (std::size_t size_);
|
||||||
|
|
||||||
|
bool open (char const *path_, char const *mode_ = "rb");
|
||||||
|
void close ();
|
||||||
|
|
||||||
|
ssize_t seek (std::size_t pos_, int origin_);
|
||||||
|
|
||||||
|
ssize_t read (void *data_, std::size_t size_);
|
||||||
|
bool readAll (void *data_, std::size_t size_);
|
||||||
|
|
||||||
|
ssize_t write (void const *data_, std::size_t size_);
|
||||||
|
bool writeAll (void const *data_, std::size_t size_);
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
T read ()
|
||||||
|
{
|
||||||
|
T data;
|
||||||
|
if (read (&data, sizeof (data)) != sizeof (data))
|
||||||
|
std::abort ();
|
||||||
|
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
void write (T const &data_)
|
||||||
|
{
|
||||||
|
if (write (&data_, sizeof (data_)) != sizeof (data_))
|
||||||
|
std::abort ();
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::unique_ptr<std::FILE, int (*) (std::FILE *)> m_fp{nullptr, nullptr};
|
||||||
|
std::unique_ptr<char[]> m_buffer;
|
||||||
|
std::size_t m_bufferSize = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
class Dir
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
~Dir ();
|
||||||
|
|
||||||
|
Dir ();
|
||||||
|
|
||||||
|
Dir (Dir const &that_) = delete;
|
||||||
|
|
||||||
|
Dir (Dir &&that_);
|
||||||
|
|
||||||
|
Dir &operator= (Dir const &that_) = delete;
|
||||||
|
|
||||||
|
Dir &operator= (Dir &&that_);
|
||||||
|
|
||||||
|
operator bool () const;
|
||||||
|
operator DIR * () const;
|
||||||
|
|
||||||
|
bool open (char const *const path_);
|
||||||
|
void close ();
|
||||||
|
struct dirent *read ();
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::unique_ptr<DIR, int (*) (DIR *)> m_dp{nullptr, nullptr};
|
||||||
|
};
|
||||||
|
}
|
@ -1,13 +0,0 @@
|
|||||||
#pragma once
|
|
||||||
|
|
||||||
/*! Loop status */
|
|
||||||
typedef enum
|
|
||||||
{
|
|
||||||
LOOP_CONTINUE, /*!< Continue looping */
|
|
||||||
LOOP_RESTART, /*!< Reinitialize */
|
|
||||||
LOOP_EXIT, /*!< Terminate looping */
|
|
||||||
} loop_status_t;
|
|
||||||
|
|
||||||
int ftp_init(void);
|
|
||||||
loop_status_t ftp_loop(void);
|
|
||||||
void ftp_exit(void);
|
|
73
include/ftpServer.h
Normal file
@ -0,0 +1,73 @@
|
|||||||
|
// ftpd is a server implementation based on the following:
|
||||||
|
// - RFC 959 (https://tools.ietf.org/html/rfc959)
|
||||||
|
// - RFC 3659 (https://tools.ietf.org/html/rfc3659)
|
||||||
|
// - suggested implementation details from https://cr.yp.to/ftp/filesystem.html
|
||||||
|
//
|
||||||
|
// Copyright (C) 2020 Michael Theall
|
||||||
|
//
|
||||||
|
// 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 <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "ftpSession.h"
|
||||||
|
#include "log.h"
|
||||||
|
#include "platform.h"
|
||||||
|
#include "socket.h"
|
||||||
|
|
||||||
|
#include <atomic>
|
||||||
|
#include <chrono>
|
||||||
|
#include <cstdint>
|
||||||
|
#include <memory>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
class FtpServer;
|
||||||
|
using UniqueFtpServer = std::unique_ptr<FtpServer>;
|
||||||
|
|
||||||
|
class FtpServer
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
~FtpServer ();
|
||||||
|
|
||||||
|
void draw ();
|
||||||
|
|
||||||
|
static UniqueFtpServer create (std::uint16_t port_);
|
||||||
|
|
||||||
|
static void updateFreeSpace ();
|
||||||
|
|
||||||
|
static std::time_t startTime ();
|
||||||
|
|
||||||
|
private:
|
||||||
|
FtpServer (std::uint16_t port_);
|
||||||
|
|
||||||
|
void handleStartButton ();
|
||||||
|
void handleStopButton ();
|
||||||
|
|
||||||
|
void loop ();
|
||||||
|
void threadFunc ();
|
||||||
|
|
||||||
|
platform::Thread m_thread;
|
||||||
|
platform::Mutex m_lock;
|
||||||
|
|
||||||
|
UniqueSocket m_socket;
|
||||||
|
|
||||||
|
std::string m_name;
|
||||||
|
|
||||||
|
SharedLog m_log;
|
||||||
|
|
||||||
|
std::vector<UniqueFtpSession> m_sessions;
|
||||||
|
|
||||||
|
std::uint16_t m_port;
|
||||||
|
|
||||||
|
std::atomic<bool> m_quit;
|
||||||
|
};
|
205
include/ftpSession.h
Normal file
@ -0,0 +1,205 @@
|
|||||||
|
// ftpd is a server implementation based on the following:
|
||||||
|
// - RFC 959 (https://tools.ietf.org/html/rfc959)
|
||||||
|
// - RFC 3659 (https://tools.ietf.org/html/rfc3659)
|
||||||
|
// - suggested implementation details from https://cr.yp.to/ftp/filesystem.html
|
||||||
|
//
|
||||||
|
// Copyright (C) 2020 Michael Theall
|
||||||
|
//
|
||||||
|
// 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 <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "fs.h"
|
||||||
|
#include "ioBuffer.h"
|
||||||
|
#include "platform.h"
|
||||||
|
#include "socket.h"
|
||||||
|
|
||||||
|
#include <chrono>
|
||||||
|
#include <memory>
|
||||||
|
#include <string_view>
|
||||||
|
#include <utility>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
class FtpSession;
|
||||||
|
using UniqueFtpSession = std::unique_ptr<FtpSession>;
|
||||||
|
|
||||||
|
class FtpSession
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
~FtpSession ();
|
||||||
|
|
||||||
|
bool dead ();
|
||||||
|
|
||||||
|
void draw ();
|
||||||
|
|
||||||
|
static UniqueFtpSession create (UniqueSocket commandSocket_);
|
||||||
|
|
||||||
|
static void poll (std::vector<UniqueFtpSession> const &sessions_);
|
||||||
|
|
||||||
|
private:
|
||||||
|
constexpr static auto COMMAND_BUFFERSIZE = 4096;
|
||||||
|
constexpr static auto RESPONSE_BUFFERSIZE = 32768;
|
||||||
|
constexpr static auto XFER_BUFFERSIZE = 65536;
|
||||||
|
constexpr static auto FILE_BUFFERSIZE = 4 * XFER_BUFFERSIZE;
|
||||||
|
|
||||||
|
#ifdef _3DS
|
||||||
|
constexpr static auto SOCK_BUFFERSIZE = 32768;
|
||||||
|
constexpr static auto POSITION_HISTORY = 100;
|
||||||
|
#else
|
||||||
|
constexpr static auto SOCK_BUFFERSIZE = XFER_BUFFERSIZE;
|
||||||
|
constexpr static auto POSITION_HISTORY = 300;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
enum class State
|
||||||
|
{
|
||||||
|
COMMAND,
|
||||||
|
DATA_CONNECT,
|
||||||
|
DATA_TRANSFER,
|
||||||
|
};
|
||||||
|
|
||||||
|
enum class XferFileMode
|
||||||
|
{
|
||||||
|
RETR,
|
||||||
|
STOR,
|
||||||
|
APPE,
|
||||||
|
};
|
||||||
|
|
||||||
|
enum class XferDirMode
|
||||||
|
{
|
||||||
|
LIST,
|
||||||
|
MLSD,
|
||||||
|
MLST,
|
||||||
|
NLST,
|
||||||
|
STAT,
|
||||||
|
};
|
||||||
|
|
||||||
|
FtpSession (UniqueSocket commandSocket_);
|
||||||
|
|
||||||
|
void setState (State state_, bool closePasv_, bool closeData_);
|
||||||
|
|
||||||
|
void closeData ();
|
||||||
|
|
||||||
|
bool changeDir (char const *args_);
|
||||||
|
|
||||||
|
bool dataAccept ();
|
||||||
|
bool dataConnect ();
|
||||||
|
|
||||||
|
void updateFreeSpace ();
|
||||||
|
|
||||||
|
int fillDirent (struct stat const &st_, std::string_view path_, char const *type_ = nullptr);
|
||||||
|
int fillDirent (std::string const &path_, char const *type_ = nullptr);
|
||||||
|
void xferFile (char const *args_, XferFileMode mode_);
|
||||||
|
void xferDir (char const *args_, XferDirMode mode_, bool workaround_);
|
||||||
|
|
||||||
|
void readCommand (int events_);
|
||||||
|
void writeResponse ();
|
||||||
|
|
||||||
|
__attribute__ ((format (printf, 2, 3))) void sendResponse (char const *fmt_, ...);
|
||||||
|
void sendResponse (std::string_view response_);
|
||||||
|
|
||||||
|
bool (FtpSession::*m_transfer) () = nullptr;
|
||||||
|
|
||||||
|
bool listTransfer ();
|
||||||
|
bool retrieveTransfer ();
|
||||||
|
bool storeTransfer ();
|
||||||
|
|
||||||
|
platform::Mutex m_lock;
|
||||||
|
|
||||||
|
SharedSocket m_commandSocket;
|
||||||
|
UniqueSocket m_pasvSocket;
|
||||||
|
SharedSocket m_dataSocket;
|
||||||
|
std::vector<SharedSocket> m_pendingCloseSocket;
|
||||||
|
|
||||||
|
IOBuffer m_commandBuffer;
|
||||||
|
IOBuffer m_responseBuffer;
|
||||||
|
IOBuffer m_xferBuffer;
|
||||||
|
|
||||||
|
SockAddr m_pasvAddr;
|
||||||
|
SockAddr m_portAddr;
|
||||||
|
|
||||||
|
std::string m_cwd = "/";
|
||||||
|
std::string m_lwd;
|
||||||
|
std::string m_rename;
|
||||||
|
std::string m_workItem;
|
||||||
|
|
||||||
|
std::string m_windowName;
|
||||||
|
std::string m_plotName;
|
||||||
|
|
||||||
|
std::uint64_t m_restartPosition = 0;
|
||||||
|
std::uint64_t m_filePosition = 0;
|
||||||
|
std::uint64_t m_fileSize = 0;
|
||||||
|
|
||||||
|
platform::steady_clock::time_point m_filePositionTime;
|
||||||
|
std::uint64_t m_filePositionHistory[POSITION_HISTORY];
|
||||||
|
float m_filePositionDeltas[POSITION_HISTORY];
|
||||||
|
float m_xferRate;
|
||||||
|
|
||||||
|
State m_state = State::COMMAND;
|
||||||
|
|
||||||
|
fs::File m_file;
|
||||||
|
fs::Dir m_dir;
|
||||||
|
|
||||||
|
XferDirMode m_xferDirMode;
|
||||||
|
|
||||||
|
bool m_pasv : 1;
|
||||||
|
bool m_port : 1;
|
||||||
|
bool m_recv : 1;
|
||||||
|
bool m_send : 1;
|
||||||
|
bool m_urgent : 1;
|
||||||
|
|
||||||
|
bool m_mlstType : 1;
|
||||||
|
bool m_mlstSize : 1;
|
||||||
|
bool m_mlstModify : 1;
|
||||||
|
bool m_mlstPerm : 1;
|
||||||
|
bool m_mlstUnixMode : 1;
|
||||||
|
|
||||||
|
void ABOR (char const *args_);
|
||||||
|
void ALLO (char const *args_);
|
||||||
|
void APPE (char const *args_);
|
||||||
|
void CDUP (char const *args_);
|
||||||
|
void CWD (char const *args_);
|
||||||
|
void DELE (char const *args_);
|
||||||
|
void FEAT (char const *args_);
|
||||||
|
void HELP (char const *args_);
|
||||||
|
void LIST (char const *args_);
|
||||||
|
void MDTM (char const *args_);
|
||||||
|
void MKD (char const *args_);
|
||||||
|
void MLSD (char const *args_);
|
||||||
|
void MLST (char const *args_);
|
||||||
|
void MODE (char const *args_);
|
||||||
|
void NLST (char const *args_);
|
||||||
|
void NOOP (char const *args_);
|
||||||
|
void OPTS (char const *args_);
|
||||||
|
void PASS (char const *args_);
|
||||||
|
void PASV (char const *args_);
|
||||||
|
void PORT (char const *args_);
|
||||||
|
void PWD (char const *args_);
|
||||||
|
void QUIT (char const *args_);
|
||||||
|
void REST (char const *args_);
|
||||||
|
void RETR (char const *args_);
|
||||||
|
void RMD (char const *args_);
|
||||||
|
void RNFR (char const *args_);
|
||||||
|
void RNTO (char const *args_);
|
||||||
|
void SIZE (char const *args_);
|
||||||
|
void STAT (char const *args_);
|
||||||
|
void STOR (char const *args_);
|
||||||
|
void STOU (char const *args_);
|
||||||
|
void STRU (char const *args_);
|
||||||
|
void SYST (char const *args_);
|
||||||
|
void TYPE (char const *args_);
|
||||||
|
void USER (char const *args_);
|
||||||
|
|
||||||
|
static std::vector<std::pair<std::string_view, void (FtpSession::*) (char const *)>> const
|
||||||
|
handlers;
|
||||||
|
};
|
1
include/imgui.h
Normal file
@ -0,0 +1 @@
|
|||||||
|
#error "Please use https://github.com/ocornut/imgui/releases/tag/v1.75"
|
51
include/ioBuffer.h
Normal file
@ -0,0 +1,51 @@
|
|||||||
|
// ftpd is a server implementation based on the following:
|
||||||
|
// - RFC 959 (https://tools.ietf.org/html/rfc959)
|
||||||
|
// - RFC 3659 (https://tools.ietf.org/html/rfc3659)
|
||||||
|
// - suggested implementation details from https://cr.yp.to/ftp/filesystem.html
|
||||||
|
//
|
||||||
|
// Copyright (C) 2020 Michael Theall
|
||||||
|
//
|
||||||
|
// 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 <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <cstddef>
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
|
class IOBuffer
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
~IOBuffer ();
|
||||||
|
|
||||||
|
IOBuffer (std::size_t size_);
|
||||||
|
|
||||||
|
char *freeArea () const;
|
||||||
|
std::size_t freeSize () const;
|
||||||
|
void markFree (std::size_t size_);
|
||||||
|
|
||||||
|
char *usedArea () const;
|
||||||
|
std::size_t usedSize () const;
|
||||||
|
void markUsed (std::size_t size_);
|
||||||
|
|
||||||
|
bool empty () const;
|
||||||
|
std::size_t capacity () const;
|
||||||
|
void clear ();
|
||||||
|
void coalesce ();
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::unique_ptr<char[]> m_buffer;
|
||||||
|
std::size_t const m_size;
|
||||||
|
std::size_t m_start = 0;
|
||||||
|
std::size_t m_end = 0;
|
||||||
|
};
|
81
include/log.h
Normal file
@ -0,0 +1,81 @@
|
|||||||
|
// ftpd is a server implementation based on the following:
|
||||||
|
// - RFC 959 (https://tools.ietf.org/html/rfc959)
|
||||||
|
// - RFC 3659 (https://tools.ietf.org/html/rfc3659)
|
||||||
|
// - suggested implementation details from https://cr.yp.to/ftp/filesystem.html
|
||||||
|
//
|
||||||
|
// Copyright (C) 2020 Michael Theall
|
||||||
|
//
|
||||||
|
// 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 <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "platform.h"
|
||||||
|
|
||||||
|
#include <cstdarg>
|
||||||
|
#include <memory>
|
||||||
|
#include <string>
|
||||||
|
#include <string_view>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
class Log;
|
||||||
|
using SharedLog = std::shared_ptr<Log>;
|
||||||
|
using WeakLog = std::weak_ptr<Log>;
|
||||||
|
|
||||||
|
class Log
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
enum Level
|
||||||
|
{
|
||||||
|
DEBUG,
|
||||||
|
INFO,
|
||||||
|
ERROR,
|
||||||
|
COMMAND,
|
||||||
|
RESPONSE,
|
||||||
|
};
|
||||||
|
|
||||||
|
~Log ();
|
||||||
|
|
||||||
|
void draw ();
|
||||||
|
|
||||||
|
static SharedLog create ();
|
||||||
|
static void bind (SharedLog log_);
|
||||||
|
|
||||||
|
__attribute__ ((format (printf, 1, 2))) static void debug (char const *fmt_, ...);
|
||||||
|
__attribute__ ((format (printf, 1, 2))) static void info (char const *fmt_, ...);
|
||||||
|
__attribute__ ((format (printf, 1, 2))) static void error (char const *fmt_, ...);
|
||||||
|
__attribute__ ((format (printf, 1, 2))) static void command (char const *fmt_, ...);
|
||||||
|
__attribute__ ((format (printf, 1, 2))) static void response (char const *fmt_, ...);
|
||||||
|
|
||||||
|
static void log (Level level_, char const *fmt_, va_list ap_);
|
||||||
|
static void log (Level level_, std::string_view message_);
|
||||||
|
|
||||||
|
private:
|
||||||
|
Log ();
|
||||||
|
|
||||||
|
void _log (Level level_, char const *fmt_, va_list ap_);
|
||||||
|
|
||||||
|
struct Message
|
||||||
|
{
|
||||||
|
Message (Level const level_, std::string message_)
|
||||||
|
: level (level_), message (std::move (message_))
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
Level level;
|
||||||
|
std::string message;
|
||||||
|
};
|
||||||
|
|
||||||
|
std::vector<Message> m_messages;
|
||||||
|
platform::Mutex m_lock;
|
||||||
|
};
|
95
include/platform.h
Normal file
@ -0,0 +1,95 @@
|
|||||||
|
// ftpd is a server implementation based on the following:
|
||||||
|
// - RFC 959 (https://tools.ietf.org/html/rfc959)
|
||||||
|
// - RFC 3659 (https://tools.ietf.org/html/rfc3659)
|
||||||
|
// - suggested implementation details from https://cr.yp.to/ftp/filesystem.html
|
||||||
|
//
|
||||||
|
// Copyright (C) 2020 Michael Theall
|
||||||
|
//
|
||||||
|
// 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 <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#ifdef _3DS
|
||||||
|
#include <3ds.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include <chrono>
|
||||||
|
#include <cstdint>
|
||||||
|
#include <functional>
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
|
namespace platform
|
||||||
|
{
|
||||||
|
bool init ();
|
||||||
|
bool loop ();
|
||||||
|
void render ();
|
||||||
|
void exit ();
|
||||||
|
|
||||||
|
#ifdef _3DS
|
||||||
|
struct steady_clock
|
||||||
|
{
|
||||||
|
using rep = std::uint64_t;
|
||||||
|
using period = std::ratio<1, SYSCLOCK_ARM11>;
|
||||||
|
using duration = std::chrono::duration<rep, period>;
|
||||||
|
using time_point = std::chrono::time_point<steady_clock>;
|
||||||
|
|
||||||
|
constexpr static bool is_steady = true;
|
||||||
|
static time_point now () noexcept
|
||||||
|
{
|
||||||
|
return time_point (duration (svcGetSystemTick ()));
|
||||||
|
}
|
||||||
|
};
|
||||||
|
#else
|
||||||
|
using steady_clock = std::chrono::steady_clock;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
class Thread
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
~Thread ();
|
||||||
|
Thread ();
|
||||||
|
|
||||||
|
Thread (std::function<void ()> func_);
|
||||||
|
|
||||||
|
Thread (Thread const &that_) = delete;
|
||||||
|
|
||||||
|
Thread (Thread &&that_);
|
||||||
|
|
||||||
|
Thread &operator= (Thread const &that_) = delete;
|
||||||
|
|
||||||
|
Thread &operator= (Thread &&that_);
|
||||||
|
|
||||||
|
void join ();
|
||||||
|
|
||||||
|
static void sleep (std::chrono::milliseconds timeout_);
|
||||||
|
|
||||||
|
private:
|
||||||
|
class privateData_t;
|
||||||
|
std::unique_ptr<privateData_t> m_d;
|
||||||
|
};
|
||||||
|
|
||||||
|
class Mutex
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
~Mutex ();
|
||||||
|
Mutex ();
|
||||||
|
|
||||||
|
void lock ();
|
||||||
|
void unlock ();
|
||||||
|
|
||||||
|
private:
|
||||||
|
class privateData_t;
|
||||||
|
std::unique_ptr<privateData_t> m_d;
|
||||||
|
};
|
||||||
|
}
|
70
include/sockAddr.h
Normal file
@ -0,0 +1,70 @@
|
|||||||
|
// ftpd is a server implementation based on the following:
|
||||||
|
// - RFC 959 (https://tools.ietf.org/html/rfc959)
|
||||||
|
// - RFC 3659 (https://tools.ietf.org/html/rfc3659)
|
||||||
|
// - suggested implementation details from https://cr.yp.to/ftp/filesystem.html
|
||||||
|
//
|
||||||
|
// Copyright (C) 2020 Michael Theall
|
||||||
|
//
|
||||||
|
// 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 <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <netinet/in.h>
|
||||||
|
#include <sys/socket.h>
|
||||||
|
|
||||||
|
#include <cstdint>
|
||||||
|
|
||||||
|
class SockAddr
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
~SockAddr ();
|
||||||
|
|
||||||
|
SockAddr ();
|
||||||
|
|
||||||
|
SockAddr (SockAddr const &that_);
|
||||||
|
|
||||||
|
SockAddr (SockAddr &&that_);
|
||||||
|
|
||||||
|
SockAddr &operator= (SockAddr const &that_);
|
||||||
|
|
||||||
|
SockAddr &operator= (SockAddr &&that_);
|
||||||
|
|
||||||
|
SockAddr (struct sockaddr const &addr_);
|
||||||
|
|
||||||
|
SockAddr (struct sockaddr_in const &addr_);
|
||||||
|
|
||||||
|
#ifndef _3DS
|
||||||
|
SockAddr (struct sockaddr_in6 const &addr_);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
SockAddr (struct sockaddr_storage const &addr_);
|
||||||
|
|
||||||
|
operator struct sockaddr_in const & () const;
|
||||||
|
|
||||||
|
#ifndef _3DS
|
||||||
|
operator struct sockaddr_in6 const & () const;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
operator struct sockaddr_storage const & () const;
|
||||||
|
|
||||||
|
operator struct sockaddr * ();
|
||||||
|
operator struct sockaddr const * () const;
|
||||||
|
|
||||||
|
std::uint16_t port () const;
|
||||||
|
char const *name (char *buffer_, std::size_t size_) const;
|
||||||
|
char const *name () const;
|
||||||
|
|
||||||
|
private:
|
||||||
|
struct sockaddr_storage m_addr = {};
|
||||||
|
};
|
97
include/socket.h
Normal file
@ -0,0 +1,97 @@
|
|||||||
|
// ftpd is a server implementation based on the following:
|
||||||
|
// - RFC 959 (https://tools.ietf.org/html/rfc959)
|
||||||
|
// - RFC 3659 (https://tools.ietf.org/html/rfc3659)
|
||||||
|
// - suggested implementation details from https://cr.yp.to/ftp/filesystem.html
|
||||||
|
//
|
||||||
|
// Copyright (C) 2020 Michael Theall
|
||||||
|
//
|
||||||
|
// 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 <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "ioBuffer.h"
|
||||||
|
#include "sockAddr.h"
|
||||||
|
|
||||||
|
#include <chrono>
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
|
class Socket;
|
||||||
|
using UniqueSocket = std::unique_ptr<Socket>;
|
||||||
|
using SharedSocket = std::shared_ptr<Socket>;
|
||||||
|
|
||||||
|
class Socket
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
struct PollInfo
|
||||||
|
{
|
||||||
|
std::reference_wrapper<Socket> socket;
|
||||||
|
int events;
|
||||||
|
int revents;
|
||||||
|
};
|
||||||
|
|
||||||
|
~Socket ();
|
||||||
|
|
||||||
|
UniqueSocket accept ();
|
||||||
|
int atMark ();
|
||||||
|
bool bind (SockAddr const &addr_);
|
||||||
|
bool connect (SockAddr const &addr_);
|
||||||
|
bool listen (int backlog_);
|
||||||
|
bool shutdown (int how_);
|
||||||
|
|
||||||
|
bool setLinger (bool enable_, std::chrono::seconds time_);
|
||||||
|
bool setNonBlocking (bool nonBlocking_ = true);
|
||||||
|
bool setReuseAddress (bool reuse_ = true);
|
||||||
|
bool setRecvBufferSize (std::size_t size_);
|
||||||
|
bool setSendBufferSize (std::size_t size_);
|
||||||
|
|
||||||
|
ssize_t read (void *buffer_, std::size_t size_, bool oob_ = false);
|
||||||
|
ssize_t read (IOBuffer &buffer_, bool oob_ = false);
|
||||||
|
ssize_t write (void const *buffer_, std::size_t size_);
|
||||||
|
ssize_t write (IOBuffer &buffer_);
|
||||||
|
|
||||||
|
SockAddr const &sockName () const;
|
||||||
|
SockAddr const &peerName () const;
|
||||||
|
|
||||||
|
static UniqueSocket create ();
|
||||||
|
|
||||||
|
static int poll (PollInfo *info_, std::size_t count_, std::chrono::milliseconds timeout_);
|
||||||
|
|
||||||
|
int fd () const
|
||||||
|
{
|
||||||
|
return m_fd;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
Socket () = delete;
|
||||||
|
|
||||||
|
Socket (int fd_);
|
||||||
|
|
||||||
|
Socket (int fd_, SockAddr const &sockName_, SockAddr const &peerName_);
|
||||||
|
|
||||||
|
Socket (Socket const &that_) = delete;
|
||||||
|
|
||||||
|
Socket (Socket &&that_) = delete;
|
||||||
|
|
||||||
|
Socket &operator= (Socket const &that_) = delete;
|
||||||
|
|
||||||
|
Socket &operator= (Socket &&that_) = delete;
|
||||||
|
|
||||||
|
SockAddr m_sockName;
|
||||||
|
SockAddr m_peerName;
|
||||||
|
|
||||||
|
int const m_fd;
|
||||||
|
|
||||||
|
bool m_listening : 1;
|
||||||
|
bool m_connected : 1;
|
||||||
|
};
|
623
source/3ds/imgui_citro3d.cpp
Normal file
@ -0,0 +1,623 @@
|
|||||||
|
// ftpd is a server implementation based on the following:
|
||||||
|
// - RFC 959 (https://tools.ietf.org/html/rfc959)
|
||||||
|
// - RFC 3659 (https://tools.ietf.org/html/rfc3659)
|
||||||
|
// - suggested implementation details from https://cr.yp.to/ftp/filesystem.html
|
||||||
|
//
|
||||||
|
// Copyright (C) 2020 Michael Theall
|
||||||
|
//
|
||||||
|
// 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 <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
#include "imgui_citro3d.h"
|
||||||
|
|
||||||
|
#include <citro3d.h>
|
||||||
|
|
||||||
|
#include "vshader_shbin.h"
|
||||||
|
|
||||||
|
#include "imgui.h"
|
||||||
|
|
||||||
|
#include <algorithm>
|
||||||
|
#include <cstdlib>
|
||||||
|
#include <cstring>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
namespace
|
||||||
|
{
|
||||||
|
std::vector<ImWchar> s_fontRanges;
|
||||||
|
|
||||||
|
constexpr auto CLEAR_COLOR = 0x204B7AFF;
|
||||||
|
|
||||||
|
constexpr auto DISPLAY_TRANSFER_FLAGS =
|
||||||
|
GX_TRANSFER_FLIP_VERT (0) | GX_TRANSFER_OUT_TILED (0) | GX_TRANSFER_RAW_COPY (0) |
|
||||||
|
GX_TRANSFER_IN_FORMAT (GX_TRANSFER_FMT_RGBA8) | GX_TRANSFER_OUT_FORMAT (GX_TRANSFER_FMT_RGB8) |
|
||||||
|
GX_TRANSFER_SCALING (GX_TRANSFER_SCALE_NO);
|
||||||
|
|
||||||
|
C3D_RenderTarget *s_top = nullptr;
|
||||||
|
C3D_RenderTarget *s_bottom = nullptr;
|
||||||
|
|
||||||
|
DVLB_s *s_vsh = nullptr;
|
||||||
|
shaderProgram_s s_program;
|
||||||
|
|
||||||
|
int s_projLocation;
|
||||||
|
C3D_Mtx s_projTop;
|
||||||
|
C3D_Mtx s_projBottom;
|
||||||
|
|
||||||
|
std::vector<C3D_Tex> s_fontTextures;
|
||||||
|
float s_textScale;
|
||||||
|
|
||||||
|
std::uint32_t s_boundScissor[4];
|
||||||
|
ImDrawVert *s_boundVtxData;
|
||||||
|
C3D_Tex *s_boundTexture;
|
||||||
|
|
||||||
|
ImDrawVert *s_vtxData = nullptr;
|
||||||
|
std::size_t s_vtxSize = 0;
|
||||||
|
ImDrawIdx *s_idxData = nullptr;
|
||||||
|
std::size_t s_idxSize = 0;
|
||||||
|
|
||||||
|
std::uint32_t fontCodePointFromGlyphIndex (CFNT_s *const font_, int const glyphIndex_)
|
||||||
|
{
|
||||||
|
for (auto cmap = fontGetInfo (font_)->cmap; cmap; cmap = cmap->next)
|
||||||
|
{
|
||||||
|
switch (cmap->mappingMethod)
|
||||||
|
{
|
||||||
|
case CMAP_TYPE_DIRECT:
|
||||||
|
assert (cmap->codeEnd >= cmap->codeBegin);
|
||||||
|
if (glyphIndex_ >= cmap->indexOffset &&
|
||||||
|
glyphIndex_ <= cmap->codeEnd - cmap->codeBegin + cmap->indexOffset)
|
||||||
|
return glyphIndex_ - cmap->indexOffset + cmap->codeBegin;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case CMAP_TYPE_TABLE:
|
||||||
|
for (int i = 0; i <= cmap->codeEnd - cmap->codeBegin; ++i)
|
||||||
|
{
|
||||||
|
if (cmap->indexTable[i] == glyphIndex_)
|
||||||
|
return cmap->codeBegin + i;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case CMAP_TYPE_SCAN:
|
||||||
|
for (unsigned i = 0; i < cmap->nScanEntries; ++i)
|
||||||
|
{
|
||||||
|
assert (cmap->scanEntries[i].code >= cmap->codeBegin);
|
||||||
|
assert (cmap->scanEntries[i].code <= cmap->codeEnd);
|
||||||
|
if (glyphIndex_ == cmap->scanEntries[i].glyphIndex)
|
||||||
|
return cmap->scanEntries[i].code;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void setupRenderState (gfxScreen_t const screen_)
|
||||||
|
{
|
||||||
|
C3D_CullFace (GPU_CULL_NONE);
|
||||||
|
|
||||||
|
// configure attributes for user with vertex shader
|
||||||
|
auto const attrInfo = C3D_GetAttrInfo ();
|
||||||
|
AttrInfo_Init (attrInfo);
|
||||||
|
AttrInfo_AddLoader (attrInfo, 0, GPU_FLOAT, 2); // v0 = inPos
|
||||||
|
AttrInfo_AddLoader (attrInfo, 1, GPU_FLOAT, 2); // v1 = inUv
|
||||||
|
AttrInfo_AddLoader (attrInfo, 2, GPU_UNSIGNED_BYTE, 4); // v2 = inColor
|
||||||
|
|
||||||
|
std::memset (s_boundScissor, 0xFF, sizeof (s_boundScissor));
|
||||||
|
s_boundVtxData = nullptr;
|
||||||
|
s_boundTexture = nullptr;
|
||||||
|
|
||||||
|
C3D_BindProgram (&s_program);
|
||||||
|
|
||||||
|
C3D_DepthTest (true, GPU_GREATER, GPU_WRITE_COLOR);
|
||||||
|
|
||||||
|
C3D_AlphaBlend (GPU_BLEND_ADD,
|
||||||
|
GPU_BLEND_ADD,
|
||||||
|
GPU_SRC_ALPHA,
|
||||||
|
GPU_ONE_MINUS_SRC_ALPHA,
|
||||||
|
GPU_SRC_ALPHA,
|
||||||
|
GPU_ONE_MINUS_SRC_ALPHA);
|
||||||
|
|
||||||
|
if (screen_ == GFX_TOP)
|
||||||
|
C3D_FVUnifMtx4x4 (GPU_VERTEX_SHADER, s_projLocation, &s_projTop);
|
||||||
|
else
|
||||||
|
C3D_FVUnifMtx4x4 (GPU_VERTEX_SHADER, s_projLocation, &s_projBottom);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void imgui::citro3d::init ()
|
||||||
|
{
|
||||||
|
// Setup back-end capabilities flags
|
||||||
|
ImGuiIO &io = ImGui::GetIO ();
|
||||||
|
|
||||||
|
io.BackendRendererName = "citro3d";
|
||||||
|
io.BackendFlags |= ImGuiBackendFlags_RendererHasVtxOffset;
|
||||||
|
|
||||||
|
C3D_Init (C3D_DEFAULT_CMDBUF_SIZE);
|
||||||
|
|
||||||
|
s_top = C3D_RenderTargetCreate (240, 400, GPU_RB_RGBA8, GPU_RB_DEPTH24_STENCIL8);
|
||||||
|
C3D_RenderTargetSetOutput (s_top, GFX_TOP, GFX_LEFT, DISPLAY_TRANSFER_FLAGS);
|
||||||
|
|
||||||
|
s_bottom = C3D_RenderTargetCreate (240, 320, GPU_RB_RGBA8, GPU_RB_DEPTH24_STENCIL8);
|
||||||
|
C3D_RenderTargetSetOutput (s_bottom, GFX_BOTTOM, GFX_LEFT, DISPLAY_TRANSFER_FLAGS);
|
||||||
|
|
||||||
|
s_vsh = DVLB_ParseFile (
|
||||||
|
const_cast<std::uint32_t *> (reinterpret_cast<std::uint32_t const *> (vshader_shbin)),
|
||||||
|
vshader_shbin_size);
|
||||||
|
shaderProgramInit (&s_program);
|
||||||
|
shaderProgramSetVsh (&s_program, &s_vsh->DVLE[0]);
|
||||||
|
|
||||||
|
s_projLocation = shaderInstanceGetUniformLocation (s_program.vertexShader, "proj");
|
||||||
|
|
||||||
|
Mtx_OrthoTilt (&s_projTop, 0.0f, 800.0f, 480.0f, 0.0f, -1.0f, 1.0f, false);
|
||||||
|
Mtx_OrthoTilt (&s_projBottom, 80.0f, 720.0f, 960.0f, 480.0f, -1.0f, 1.0f, false);
|
||||||
|
|
||||||
|
s_vtxSize = 65536;
|
||||||
|
s_vtxData = reinterpret_cast<ImDrawVert *> (linearAlloc (sizeof (ImDrawVert) * s_vtxSize));
|
||||||
|
if (!s_vtxData)
|
||||||
|
svcBreak (USERBREAK_PANIC);
|
||||||
|
|
||||||
|
s_idxSize = 65536;
|
||||||
|
s_idxData = reinterpret_cast<ImDrawIdx *> (linearAlloc (sizeof (ImDrawIdx) * s_idxSize));
|
||||||
|
if (!s_idxData)
|
||||||
|
svcBreak (USERBREAK_PANIC);
|
||||||
|
|
||||||
|
// ensure the shared system font is mapped
|
||||||
|
if (R_FAILED (fontEnsureMapped ()))
|
||||||
|
svcBreak (USERBREAK_PANIC);
|
||||||
|
|
||||||
|
// load the glyph texture sheets
|
||||||
|
auto const font = fontGetSystemFont ();
|
||||||
|
auto const fontInfo = fontGetInfo (font);
|
||||||
|
auto const glyphInfo = fontGetGlyphInfo (font);
|
||||||
|
assert (s_fontTextures.empty ());
|
||||||
|
s_fontTextures.resize (glyphInfo->nSheets + 1);
|
||||||
|
std::memset (s_fontTextures.data (), 0x00, s_fontTextures.size () * sizeof (s_fontTextures[0]));
|
||||||
|
|
||||||
|
s_textScale = 30.0f / glyphInfo->cellHeight;
|
||||||
|
|
||||||
|
for (unsigned i = 0; i < glyphInfo->nSheets; ++i)
|
||||||
|
{
|
||||||
|
auto &tex = s_fontTextures[i];
|
||||||
|
tex.data = fontGetGlyphSheetTex (font, i);
|
||||||
|
if (!tex.data)
|
||||||
|
svcBreak (USERBREAK_PANIC);
|
||||||
|
tex.fmt = static_cast<GPU_TEXCOLOR> (glyphInfo->sheetFmt);
|
||||||
|
tex.size = glyphInfo->sheetSize;
|
||||||
|
tex.width = glyphInfo->sheetWidth;
|
||||||
|
tex.height = glyphInfo->sheetHeight;
|
||||||
|
tex.param = GPU_TEXTURE_MAG_FILTER (GPU_LINEAR) | GPU_TEXTURE_MIN_FILTER (GPU_LINEAR) |
|
||||||
|
GPU_TEXTURE_WRAP_S (GPU_REPEAT) | GPU_TEXTURE_WRAP_T (GPU_REPEAT);
|
||||||
|
tex.border = 0xFFFFFFFF;
|
||||||
|
tex.lodParam = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
auto &tex = s_fontTextures[glyphInfo->nSheets];
|
||||||
|
C3D_TexInit (&tex, 8, 8, GPU_A4);
|
||||||
|
|
||||||
|
std::uint32_t size;
|
||||||
|
auto data = C3D_Tex2DGetImagePtr (&tex, 0, &size);
|
||||||
|
if (!data || !size)
|
||||||
|
svcBreak (USERBREAK_PANIC);
|
||||||
|
std::memset (data, 0xFF, size);
|
||||||
|
}
|
||||||
|
|
||||||
|
ImWchar alterChar = fontCodePointFromGlyphIndex (font, fontInfo->alterCharIndex);
|
||||||
|
if (!alterChar)
|
||||||
|
alterChar = '?';
|
||||||
|
|
||||||
|
std::vector<ImWchar> charSet;
|
||||||
|
for (auto cmap = fontInfo->cmap; cmap; cmap = cmap->next)
|
||||||
|
{
|
||||||
|
switch (cmap->mappingMethod)
|
||||||
|
{
|
||||||
|
case CMAP_TYPE_DIRECT:
|
||||||
|
case CMAP_TYPE_TABLE:
|
||||||
|
assert (cmap->codeEnd >= cmap->codeBegin);
|
||||||
|
charSet.reserve (charSet.size () + cmap->codeEnd - cmap->codeBegin + 1);
|
||||||
|
for (auto i = cmap->codeBegin; i <= cmap->codeEnd; ++i)
|
||||||
|
charSet.emplace_back (i);
|
||||||
|
break;
|
||||||
|
case CMAP_TYPE_SCAN:
|
||||||
|
charSet.reserve (charSet.size () + cmap->nScanEntries);
|
||||||
|
for (unsigned i = 0; i < cmap->nScanEntries; ++i)
|
||||||
|
{
|
||||||
|
assert (cmap->scanEntries[i].code >= cmap->codeBegin);
|
||||||
|
assert (cmap->scanEntries[i].code <= cmap->codeEnd);
|
||||||
|
charSet.emplace_back (cmap->scanEntries[i].code);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (charSet.empty ())
|
||||||
|
svcBreak (USERBREAK_PANIC);
|
||||||
|
|
||||||
|
std::sort (std::begin (charSet), std::end (charSet));
|
||||||
|
charSet.erase (std::unique (std::begin (charSet), std::end (charSet)), std::end (charSet));
|
||||||
|
|
||||||
|
auto it = std::begin (charSet);
|
||||||
|
ImWchar start = *it++;
|
||||||
|
ImWchar prev = start;
|
||||||
|
while (it != std::end (charSet))
|
||||||
|
{
|
||||||
|
if (*it != prev + 1)
|
||||||
|
{
|
||||||
|
s_fontRanges.emplace_back (start);
|
||||||
|
s_fontRanges.emplace_back (prev);
|
||||||
|
|
||||||
|
start = *it;
|
||||||
|
}
|
||||||
|
|
||||||
|
prev = *it++;
|
||||||
|
}
|
||||||
|
s_fontRanges.emplace_back (start);
|
||||||
|
s_fontRanges.emplace_back (prev);
|
||||||
|
s_fontRanges.emplace_back (0);
|
||||||
|
|
||||||
|
auto const atlas = ImGui::GetIO ().Fonts;
|
||||||
|
atlas->Clear ();
|
||||||
|
atlas->TexWidth = glyphInfo->sheetWidth;
|
||||||
|
atlas->TexHeight = glyphInfo->sheetHeight * glyphInfo->nSheets;
|
||||||
|
atlas->TexUvScale = ImVec2 (1.0f / atlas->TexWidth, 1.0f / atlas->TexHeight);
|
||||||
|
atlas->TexUvWhitePixel = ImVec2 (0.5f / 8.0f, glyphInfo->nSheets + 0.5f / 8.0f);
|
||||||
|
atlas->TexPixelsAlpha8 = static_cast<unsigned char *> (IM_ALLOC (1)); // dummy allocation
|
||||||
|
|
||||||
|
ImFontConfig config;
|
||||||
|
config.FontData = nullptr;
|
||||||
|
config.FontDataSize = 0;
|
||||||
|
config.FontDataOwnedByAtlas = true;
|
||||||
|
config.FontNo = 0;
|
||||||
|
config.SizePixels = 14.0f;
|
||||||
|
config.OversampleH = 3;
|
||||||
|
config.OversampleV = 1;
|
||||||
|
config.PixelSnapH = false;
|
||||||
|
config.GlyphExtraSpacing = ImVec2 (0.0f, 0.0f);
|
||||||
|
config.GlyphOffset = ImVec2 (0.0f, 0.0f);
|
||||||
|
config.GlyphRanges = s_fontRanges.data ();
|
||||||
|
config.GlyphMinAdvanceX = 0.0f;
|
||||||
|
config.GlyphMaxAdvanceX = std::numeric_limits<float>::max ();
|
||||||
|
config.MergeMode = false;
|
||||||
|
config.RasterizerFlags = 0;
|
||||||
|
config.RasterizerMultiply = 1.0f;
|
||||||
|
config.EllipsisChar = 0x2026;
|
||||||
|
std::memset (config.Name, 0, sizeof (config.Name));
|
||||||
|
|
||||||
|
auto const imFont = IM_NEW (ImFont);
|
||||||
|
config.DstFont = imFont;
|
||||||
|
|
||||||
|
atlas->ConfigData.push_back (config);
|
||||||
|
atlas->Fonts.push_back (imFont);
|
||||||
|
// atlas->CustomRectIds[0] = atlas->AddCustomRectRegular (0x80000000, 108 * 2 + 1, 27);
|
||||||
|
// atlas->CustomRects[0].X = 0;
|
||||||
|
// atlas->CustomRects[0].Y = 0;
|
||||||
|
atlas->SetTexID (s_fontTextures.data ());
|
||||||
|
|
||||||
|
imFont->FallbackAdvanceX = fontInfo->defaultWidth.charWidth;
|
||||||
|
imFont->FontSize = fontInfo->lineFeed;
|
||||||
|
|
||||||
|
fontGlyphPos_s glyphPos;
|
||||||
|
for (auto const &code : charSet)
|
||||||
|
{
|
||||||
|
auto const glyphIndex = fontGlyphIndexFromCodePoint (font, code);
|
||||||
|
if (glyphIndex < 0)
|
||||||
|
svcBreak (USERBREAK_PANIC);
|
||||||
|
|
||||||
|
fontCalcGlyphPos (&glyphPos,
|
||||||
|
font,
|
||||||
|
glyphIndex,
|
||||||
|
GLYPH_POS_CALC_VTXCOORD | GLYPH_POS_AT_BASELINE,
|
||||||
|
1.0f,
|
||||||
|
1.0f);
|
||||||
|
|
||||||
|
ImFontGlyph glyph;
|
||||||
|
|
||||||
|
glyph.Codepoint = code;
|
||||||
|
glyph.AdvanceX = glyphPos.xAdvance;
|
||||||
|
glyph.X0 = glyphPos.vtxcoord.left;
|
||||||
|
glyph.Y0 = glyphPos.vtxcoord.top;
|
||||||
|
glyph.X1 = glyphPos.vtxcoord.right;
|
||||||
|
glyph.Y1 = glyphPos.vtxcoord.bottom;
|
||||||
|
glyph.U0 = glyphPos.texcoord.left;
|
||||||
|
glyph.V0 = glyphPos.sheetIndex + glyphPos.texcoord.top;
|
||||||
|
glyph.U1 = glyphPos.texcoord.right;
|
||||||
|
glyph.V1 = glyphPos.sheetIndex + glyphPos.texcoord.bottom;
|
||||||
|
|
||||||
|
imFont->Glyphs.push_back (glyph);
|
||||||
|
imFont->MetricsTotalSurface +=
|
||||||
|
static_cast<int> ((glyph.U1 - glyph.U0) * atlas->TexWidth + 1.99f) *
|
||||||
|
static_cast<int> ((glyph.V1 - glyph.V0) * atlas->TexHeight + 1.99f);
|
||||||
|
}
|
||||||
|
|
||||||
|
imFont->BuildLookupTable ();
|
||||||
|
|
||||||
|
imFont->DisplayOffset.x = 0.0f;
|
||||||
|
imFont->DisplayOffset.y = fontInfo->ascent;
|
||||||
|
|
||||||
|
imFont->ContainerAtlas = atlas;
|
||||||
|
imFont->ConfigData = &atlas->ConfigData[0];
|
||||||
|
imFont->ConfigDataCount = 1;
|
||||||
|
imFont->FallbackChar = alterChar;
|
||||||
|
imFont->EllipsisChar = config.EllipsisChar;
|
||||||
|
imFont->Scale = 1.0f;
|
||||||
|
imFont->Ascent = fontInfo->ascent;
|
||||||
|
imFont->Descent = 0.0f;
|
||||||
|
}
|
||||||
|
|
||||||
|
void imgui::citro3d::exit ()
|
||||||
|
{
|
||||||
|
linearFree (s_idxData);
|
||||||
|
linearFree (s_vtxData);
|
||||||
|
|
||||||
|
assert (!s_fontTextures.empty ());
|
||||||
|
C3D_TexDelete (&s_fontTextures.back ());
|
||||||
|
|
||||||
|
shaderProgramFree (&s_program);
|
||||||
|
DVLB_Free (s_vsh);
|
||||||
|
|
||||||
|
C3D_RenderTargetDelete (s_bottom);
|
||||||
|
C3D_RenderTargetDelete (s_top);
|
||||||
|
|
||||||
|
C3D_Fini ();
|
||||||
|
}
|
||||||
|
|
||||||
|
void imgui::citro3d::newFrame ()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
void imgui::citro3d::render ()
|
||||||
|
{
|
||||||
|
C3D_FrameBegin (C3D_FRAME_SYNCDRAW);
|
||||||
|
C3D_RenderTargetClear (s_top, C3D_CLEAR_ALL, CLEAR_COLOR, 0);
|
||||||
|
C3D_RenderTargetClear (s_bottom, C3D_CLEAR_ALL, CLEAR_COLOR, 0);
|
||||||
|
|
||||||
|
auto const drawData = ImGui::GetDrawData ();
|
||||||
|
if (drawData->CmdListsCount <= 0)
|
||||||
|
{
|
||||||
|
C3D_FrameEnd (0);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
unsigned width = drawData->DisplaySize.x * drawData->FramebufferScale.x;
|
||||||
|
unsigned height = drawData->DisplaySize.y * drawData->FramebufferScale.y;
|
||||||
|
if (width <= 0 || height <= 0)
|
||||||
|
{
|
||||||
|
C3D_FrameEnd (0);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (s_vtxSize < static_cast<std::size_t> (drawData->TotalVtxCount))
|
||||||
|
{
|
||||||
|
linearFree (s_vtxData);
|
||||||
|
|
||||||
|
// add 10% to avoid growing many frames in a row
|
||||||
|
s_vtxSize = drawData->TotalVtxCount * 1.1f;
|
||||||
|
s_vtxData = reinterpret_cast<ImDrawVert *> (linearAlloc (sizeof (ImDrawVert) * s_vtxSize));
|
||||||
|
if (!s_vtxData)
|
||||||
|
svcBreak (USERBREAK_PANIC);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (s_idxSize < static_cast<std::size_t> (drawData->TotalIdxCount))
|
||||||
|
{
|
||||||
|
// add 10% to avoid growing many frames in a row
|
||||||
|
s_idxSize = drawData->TotalIdxCount * 1.1f;
|
||||||
|
s_idxData = reinterpret_cast<ImDrawIdx *> (linearAlloc (sizeof (ImDrawIdx) * s_idxSize));
|
||||||
|
if (!s_vtxData)
|
||||||
|
svcBreak (USERBREAK_PANIC);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Will project scissor/clipping rectangles into framebuffer space
|
||||||
|
// (0,0) unless using multi-viewports
|
||||||
|
auto const clipOff = drawData->DisplayPos;
|
||||||
|
// (1,1) unless using retina display which are often (2,2)
|
||||||
|
auto const clipScale = drawData->FramebufferScale;
|
||||||
|
|
||||||
|
// copy data into vertex/index buffers
|
||||||
|
std::size_t offsetVtx = 0;
|
||||||
|
std::size_t offsetIdx = 0;
|
||||||
|
for (int i = 0; i < drawData->CmdListsCount; ++i)
|
||||||
|
{
|
||||||
|
auto const &cmdList = *drawData->CmdLists[i];
|
||||||
|
if (s_vtxSize - offsetVtx < static_cast<std::size_t> (cmdList.VtxBuffer.Size))
|
||||||
|
svcBreak (USERBREAK_PANIC);
|
||||||
|
if (s_idxSize - offsetIdx < static_cast<std::size_t> (cmdList.IdxBuffer.Size))
|
||||||
|
svcBreak (USERBREAK_PANIC);
|
||||||
|
|
||||||
|
std::memcpy (&s_vtxData[offsetVtx],
|
||||||
|
cmdList.VtxBuffer.Data,
|
||||||
|
sizeof (ImDrawVert) * cmdList.VtxBuffer.Size);
|
||||||
|
std::memcpy (&s_idxData[offsetIdx],
|
||||||
|
cmdList.IdxBuffer.Data,
|
||||||
|
sizeof (ImDrawIdx) * cmdList.IdxBuffer.Size);
|
||||||
|
|
||||||
|
offsetVtx += cmdList.VtxBuffer.Size;
|
||||||
|
offsetIdx += cmdList.IdxBuffer.Size;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (auto const &screen : {GFX_TOP, GFX_BOTTOM})
|
||||||
|
{
|
||||||
|
if (screen == GFX_TOP)
|
||||||
|
C3D_FrameDrawOn (s_top);
|
||||||
|
else
|
||||||
|
C3D_FrameDrawOn (s_bottom);
|
||||||
|
|
||||||
|
setupRenderState (screen);
|
||||||
|
|
||||||
|
offsetVtx = 0;
|
||||||
|
offsetIdx = 0;
|
||||||
|
|
||||||
|
// Render command lists
|
||||||
|
for (int i = 0; i < drawData->CmdListsCount; ++i)
|
||||||
|
{
|
||||||
|
auto const &cmdList = *drawData->CmdLists[i];
|
||||||
|
for (auto const &cmd : cmdList.CmdBuffer)
|
||||||
|
{
|
||||||
|
if (cmd.UserCallback)
|
||||||
|
{
|
||||||
|
// User callback, registered via ImDrawList::AddCallback()
|
||||||
|
// (ImDrawCallback_ResetRenderState is a special callback value used by the user
|
||||||
|
// to request the renderer to reset render state.)
|
||||||
|
if (cmd.UserCallback == ImDrawCallback_ResetRenderState)
|
||||||
|
setupRenderState (screen);
|
||||||
|
else
|
||||||
|
cmd.UserCallback (&cmdList, &cmd);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// Project scissor/clipping rectangles into framebuffer space
|
||||||
|
ImVec4 clip;
|
||||||
|
clip.x = (cmd.ClipRect.x - clipOff.x) * clipScale.x;
|
||||||
|
clip.y = (cmd.ClipRect.y - clipOff.y) * clipScale.y;
|
||||||
|
clip.z = (cmd.ClipRect.z - clipOff.x) * clipScale.x;
|
||||||
|
clip.w = (cmd.ClipRect.w - clipOff.y) * clipScale.y;
|
||||||
|
|
||||||
|
if (clip.x >= width || clip.y >= height || clip.z < 0.0f || clip.w < 0.0f)
|
||||||
|
continue;
|
||||||
|
if (clip.x < 0.0f)
|
||||||
|
clip.x = 0.0f;
|
||||||
|
if (clip.y < 0.0f)
|
||||||
|
clip.y = 0.0f;
|
||||||
|
|
||||||
|
if (screen == GFX_TOP)
|
||||||
|
{
|
||||||
|
// check if clip starts on bottom screen
|
||||||
|
if (clip.y > 240.0f)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
auto const x1 = std::clamp<unsigned> (240.0f - clip.w, 0, 240);
|
||||||
|
auto const y1 = std::clamp<unsigned> (400.0f - clip.z, 0, 400);
|
||||||
|
auto const x2 = std::clamp<unsigned> (240.0f - clip.y, 0, 240);
|
||||||
|
auto const y2 = std::clamp<unsigned> (400.0f - clip.x, 0, 400);
|
||||||
|
|
||||||
|
C3D_SetScissor (GPU_SCISSOR_NORMAL, x1, y1, x2, y2);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// check if clip ends on top screen
|
||||||
|
if (clip.w < 240.0f)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
// check if clip ends before left edge of bottom screen
|
||||||
|
if (clip.z < 40.0f)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
// check if clip starts after right edge of bottom screen
|
||||||
|
if (clip.x > 360.0f)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
auto const x1 = std::clamp<unsigned> (480.0f - clip.w, 0, 240);
|
||||||
|
auto const y1 = std::clamp<unsigned> (360.0f - clip.z, 0, 320);
|
||||||
|
auto const x2 = std::clamp<unsigned> (480.0f - clip.y, 0, 240);
|
||||||
|
auto const y2 = std::clamp<unsigned> (360.0f - clip.x, 0, 320);
|
||||||
|
|
||||||
|
if (s_boundScissor[0] != x1 || s_boundScissor[1] != y1 ||
|
||||||
|
s_boundScissor[2] != x2 || s_boundScissor[3] != y2)
|
||||||
|
{
|
||||||
|
s_boundScissor[0] = x1;
|
||||||
|
s_boundScissor[1] = y1;
|
||||||
|
s_boundScissor[2] = x2;
|
||||||
|
s_boundScissor[3] = y2;
|
||||||
|
C3D_SetScissor (GPU_SCISSOR_NORMAL, x1, y1, x2, y2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
auto const vtxData = &s_vtxData[cmd.VtxOffset + offsetVtx];
|
||||||
|
if (vtxData != s_boundVtxData)
|
||||||
|
{
|
||||||
|
s_boundVtxData = &s_vtxData[cmd.VtxOffset + offsetVtx];
|
||||||
|
auto const bufInfo = C3D_GetBufInfo ();
|
||||||
|
BufInfo_Init (bufInfo);
|
||||||
|
BufInfo_Add (bufInfo, s_boundVtxData, sizeof (ImDrawVert), 3, 0x210);
|
||||||
|
}
|
||||||
|
|
||||||
|
auto tex = static_cast<C3D_Tex *> (cmd.TextureId);
|
||||||
|
if (tex == s_fontTextures.data ())
|
||||||
|
{
|
||||||
|
assert (cmd.ElemCount % 3 == 0);
|
||||||
|
|
||||||
|
// TODO get by idx not consecutive vtx
|
||||||
|
auto const getSheet = [] (auto const vtx_, auto const idx_) {
|
||||||
|
unsigned const sheet = std::min (
|
||||||
|
{vtx_[idx_[0]].uv.y, vtx_[idx_[1]].uv.y, vtx_[idx_[2]].uv.y});
|
||||||
|
for (unsigned i = 0; i < 3; ++i)
|
||||||
|
assert (vtx_[idx_[i]].uv.y - sheet <= 1.0f);
|
||||||
|
return sheet;
|
||||||
|
};
|
||||||
|
|
||||||
|
unsigned boundSheet = getSheet (&s_vtxData[cmd.VtxOffset + offsetVtx],
|
||||||
|
&s_idxData[cmd.IdxOffset + offsetIdx]);
|
||||||
|
|
||||||
|
unsigned offset = 0;
|
||||||
|
|
||||||
|
C3D_TexBind (0, &s_fontTextures[boundSheet]);
|
||||||
|
|
||||||
|
auto const env = C3D_GetTexEnv (0);
|
||||||
|
C3D_TexEnvInit (env);
|
||||||
|
C3D_TexEnvSrc (
|
||||||
|
env, C3D_RGB, GPU_PRIMARY_COLOR, GPU_PRIMARY_COLOR, GPU_PRIMARY_COLOR);
|
||||||
|
C3D_TexEnvFunc (env, C3D_RGB, GPU_REPLACE);
|
||||||
|
C3D_TexEnvSrc (
|
||||||
|
env, C3D_Alpha, GPU_TEXTURE0, GPU_PRIMARY_COLOR, GPU_PRIMARY_COLOR);
|
||||||
|
C3D_TexEnvFunc (env, C3D_Alpha, GPU_MODULATE);
|
||||||
|
|
||||||
|
for (unsigned i = 3; i < cmd.ElemCount; i += 3)
|
||||||
|
{
|
||||||
|
unsigned const sheet = getSheet (&s_vtxData[cmd.VtxOffset + offsetVtx],
|
||||||
|
&s_idxData[cmd.IdxOffset + offsetIdx + i]);
|
||||||
|
if (boundSheet != sheet)
|
||||||
|
{
|
||||||
|
C3D_DrawElements (GPU_TRIANGLES,
|
||||||
|
i - offset,
|
||||||
|
C3D_UNSIGNED_SHORT,
|
||||||
|
&s_idxData[cmd.IdxOffset + offsetIdx + offset]);
|
||||||
|
|
||||||
|
boundSheet = sheet;
|
||||||
|
offset = i;
|
||||||
|
C3D_TexBind (0, &s_fontTextures[boundSheet]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
assert ((cmd.ElemCount - offset) % 3 == 0);
|
||||||
|
C3D_DrawElements (GPU_TRIANGLES,
|
||||||
|
cmd.ElemCount - offset,
|
||||||
|
C3D_UNSIGNED_SHORT,
|
||||||
|
&s_idxData[cmd.IdxOffset + offsetIdx + offset]);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (tex != s_boundTexture)
|
||||||
|
{
|
||||||
|
C3D_TexBind (0, tex);
|
||||||
|
auto const env = C3D_GetTexEnv (0);
|
||||||
|
C3D_TexEnvInit (env);
|
||||||
|
C3D_TexEnvSrc (
|
||||||
|
env, C3D_Both, GPU_TEXTURE0, GPU_PRIMARY_COLOR, GPU_PRIMARY_COLOR);
|
||||||
|
C3D_TexEnvFunc (env, C3D_Both, GPU_MODULATE);
|
||||||
|
}
|
||||||
|
|
||||||
|
C3D_DrawElements (GPU_TRIANGLES,
|
||||||
|
cmd.ElemCount,
|
||||||
|
C3D_UNSIGNED_SHORT,
|
||||||
|
&s_idxData[cmd.IdxOffset + offsetIdx]);
|
||||||
|
}
|
||||||
|
|
||||||
|
s_boundTexture = tex;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
offsetVtx += cmdList.VtxBuffer.Size;
|
||||||
|
offsetIdx += cmdList.IdxBuffer.Size;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
C3D_FrameEnd (0);
|
||||||
|
}
|
33
source/3ds/imgui_citro3d.h
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
// ftpd is a server implementation based on the following:
|
||||||
|
// - RFC 959 (https://tools.ietf.org/html/rfc959)
|
||||||
|
// - RFC 3659 (https://tools.ietf.org/html/rfc3659)
|
||||||
|
// - suggested implementation details from https://cr.yp.to/ftp/filesystem.html
|
||||||
|
//
|
||||||
|
// Copyright (C) 2020 Michael Theall
|
||||||
|
//
|
||||||
|
// 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 <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
namespace imgui
|
||||||
|
{
|
||||||
|
namespace citro3d
|
||||||
|
{
|
||||||
|
void init ();
|
||||||
|
void exit ();
|
||||||
|
|
||||||
|
void newFrame ();
|
||||||
|
void render ();
|
||||||
|
}
|
||||||
|
}
|
163
source/3ds/imgui_ctru.cpp
Normal file
@ -0,0 +1,163 @@
|
|||||||
|
// ftpd is a server implementation based on the following:
|
||||||
|
// - RFC 959 (https://tools.ietf.org/html/rfc959)
|
||||||
|
// - RFC 3659 (https://tools.ietf.org/html/rfc3659)
|
||||||
|
// - suggested implementation details from https://cr.yp.to/ftp/filesystem.html
|
||||||
|
//
|
||||||
|
// Copyright (C) 2020 Michael Theall
|
||||||
|
//
|
||||||
|
// 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 <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
#include "imgui_ctru.h"
|
||||||
|
|
||||||
|
#include "imgui.h"
|
||||||
|
|
||||||
|
#include "fs.h"
|
||||||
|
|
||||||
|
#include <3ds.h>
|
||||||
|
|
||||||
|
#include <chrono>
|
||||||
|
#include <cstring>
|
||||||
|
#include <functional>
|
||||||
|
#include <string>
|
||||||
|
#include <tuple>
|
||||||
|
using namespace std::chrono_literals;
|
||||||
|
|
||||||
|
namespace
|
||||||
|
{
|
||||||
|
constexpr auto SCREEN_WIDTH = 400.0f;
|
||||||
|
constexpr auto SCREEN_HEIGHT = 480.0f;
|
||||||
|
|
||||||
|
std::string s_clipboard;
|
||||||
|
|
||||||
|
char const *getClipboardText (void *const userData_)
|
||||||
|
{
|
||||||
|
(void)userData_;
|
||||||
|
return s_clipboard.c_str ();
|
||||||
|
}
|
||||||
|
|
||||||
|
void setClipboardText (void *const userData_, char const *const text_)
|
||||||
|
{
|
||||||
|
(void)userData_;
|
||||||
|
s_clipboard = text_;
|
||||||
|
}
|
||||||
|
|
||||||
|
void updateTouch (ImGuiIO &io_)
|
||||||
|
{
|
||||||
|
if (!((hidKeysDown () | hidKeysHeld ()) & KEY_TOUCH))
|
||||||
|
{
|
||||||
|
io_.MousePos = ImVec2 (-10.0f, -10.0f);
|
||||||
|
io_.MouseDown[0] = false;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
touchPosition pos;
|
||||||
|
hidTouchRead (&pos);
|
||||||
|
|
||||||
|
// transform to bottom-screen space
|
||||||
|
io_.MousePos = ImVec2 ((pos.px + 40.0f) * 2.0f, (pos.py + 240.0f) * 2.0f);
|
||||||
|
io_.MouseDown[0] = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void updateGamepads (ImGuiIO &io_)
|
||||||
|
{
|
||||||
|
std::memset (io_.NavInputs, 0, sizeof (io_.NavInputs));
|
||||||
|
|
||||||
|
auto const buttonMapping = {
|
||||||
|
std::make_pair (KEY_A, ImGuiNavInput_Activate),
|
||||||
|
std::make_pair (KEY_B, ImGuiNavInput_Cancel),
|
||||||
|
std::make_pair (KEY_X, ImGuiNavInput_Input),
|
||||||
|
std::make_pair (KEY_Y, ImGuiNavInput_Menu),
|
||||||
|
std::make_pair (KEY_L, ImGuiNavInput_FocusPrev),
|
||||||
|
std::make_pair (KEY_L, ImGuiNavInput_TweakSlow),
|
||||||
|
std::make_pair (KEY_R, ImGuiNavInput_FocusNext),
|
||||||
|
std::make_pair (KEY_R, ImGuiNavInput_TweakFast),
|
||||||
|
std::make_pair (KEY_DUP, ImGuiNavInput_DpadUp),
|
||||||
|
std::make_pair (KEY_DRIGHT, ImGuiNavInput_DpadRight),
|
||||||
|
std::make_pair (KEY_DDOWN, ImGuiNavInput_DpadDown),
|
||||||
|
std::make_pair (KEY_DLEFT, ImGuiNavInput_DpadLeft),
|
||||||
|
};
|
||||||
|
|
||||||
|
auto const keys = hidKeysHeld ();
|
||||||
|
for (auto const &[in, out] : buttonMapping)
|
||||||
|
{
|
||||||
|
if (keys & in)
|
||||||
|
io_.NavInputs[out] = 1.0f;
|
||||||
|
}
|
||||||
|
|
||||||
|
circlePosition cpad;
|
||||||
|
auto const analogMapping = {
|
||||||
|
std::make_tuple (std::ref (cpad.dx), ImGuiNavInput_LStickLeft, -0.3f, -0.9f),
|
||||||
|
std::make_tuple (std::ref (cpad.dx), ImGuiNavInput_LStickRight, +0.3f, +0.9f),
|
||||||
|
std::make_tuple (std::ref (cpad.dy), ImGuiNavInput_LStickUp, +0.3f, +0.9f),
|
||||||
|
std::make_tuple (std::ref (cpad.dy), ImGuiNavInput_LStickDown, -0.3f, -0.9f),
|
||||||
|
};
|
||||||
|
|
||||||
|
hidCircleRead (&cpad);
|
||||||
|
for (auto const &[in, out, min, max] : analogMapping)
|
||||||
|
{
|
||||||
|
auto const value = in / static_cast<float> (0x9C);
|
||||||
|
auto const v = std::min (1.0f, (value - min) / (max - min));
|
||||||
|
io_.NavInputs[out] = std::max (io_.NavInputs[out], v);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool imgui::ctru::init ()
|
||||||
|
{
|
||||||
|
ImGuiIO &io = ImGui::GetIO ();
|
||||||
|
|
||||||
|
io.IniFilename = nullptr;
|
||||||
|
|
||||||
|
io.ConfigFlags |= ImGuiConfigFlags_IsTouchScreen;
|
||||||
|
io.ConfigFlags |= ImGuiConfigFlags_NavEnableGamepad;
|
||||||
|
|
||||||
|
io.BackendFlags |= ImGuiBackendFlags_HasGamepad;
|
||||||
|
|
||||||
|
io.BackendPlatformName = "3ds";
|
||||||
|
|
||||||
|
io.MouseDrawCursor = false;
|
||||||
|
|
||||||
|
io.SetClipboardTextFn = setClipboardText;
|
||||||
|
io.GetClipboardTextFn = getClipboardText;
|
||||||
|
io.ClipboardUserData = nullptr;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void imgui::ctru::newFrame ()
|
||||||
|
{
|
||||||
|
ImGuiIO &io = ImGui::GetIO ();
|
||||||
|
|
||||||
|
IM_ASSERT (io.Fonts->IsBuilt () &&
|
||||||
|
"Font atlas not built! It is generally built by the renderer back-end. Missing call "
|
||||||
|
"to renderer _NewFrame() function?");
|
||||||
|
|
||||||
|
io.DisplaySize = ImVec2 (SCREEN_WIDTH * 2.0f, SCREEN_HEIGHT * 2.0f);
|
||||||
|
io.DisplayFramebufferScale = ImVec2 (0.5f, 0.5f);
|
||||||
|
|
||||||
|
// Setup time step
|
||||||
|
static auto const start = svcGetSystemTick ();
|
||||||
|
static auto prev = start;
|
||||||
|
auto const now = svcGetSystemTick ();
|
||||||
|
|
||||||
|
io.DeltaTime = (now - prev) / static_cast<float> (SYSCLOCK_ARM11);
|
||||||
|
prev = now;
|
||||||
|
|
||||||
|
updateTouch (io);
|
||||||
|
updateGamepads (io);
|
||||||
|
}
|
||||||
|
|
||||||
|
void imgui::ctru::exit ()
|
||||||
|
{
|
||||||
|
}
|
32
source/3ds/imgui_ctru.h
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
// ftpd is a server implementation based on the following:
|
||||||
|
// - RFC 959 (https://tools.ietf.org/html/rfc959)
|
||||||
|
// - RFC 3659 (https://tools.ietf.org/html/rfc3659)
|
||||||
|
// - suggested implementation details from https://cr.yp.to/ftp/filesystem.html
|
||||||
|
//
|
||||||
|
// Copyright (C) 2020 Michael Theall
|
||||||
|
//
|
||||||
|
// 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 <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
namespace imgui
|
||||||
|
{
|
||||||
|
namespace ctru
|
||||||
|
{
|
||||||
|
bool init ();
|
||||||
|
void exit ();
|
||||||
|
|
||||||
|
void newFrame ();
|
||||||
|
}
|
||||||
|
}
|
348
source/3ds/platform.cpp
Normal file
@ -0,0 +1,348 @@
|
|||||||
|
// ftpd is a server implementation based on the following:
|
||||||
|
// - RFC 959 (https://tools.ietf.org/html/rfc959)
|
||||||
|
// - RFC 3659 (https://tools.ietf.org/html/rfc3659)
|
||||||
|
// - suggested implementation details from https://cr.yp.to/ftp/filesystem.html
|
||||||
|
//
|
||||||
|
// Copyright (C) 2020 Michael Theall
|
||||||
|
//
|
||||||
|
// 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 <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
#include "platform.h"
|
||||||
|
|
||||||
|
#include "fs.h"
|
||||||
|
#include "log.h"
|
||||||
|
|
||||||
|
#include "imgui_citro3d.h"
|
||||||
|
#include "imgui_ctru.h"
|
||||||
|
|
||||||
|
#include "imgui.h"
|
||||||
|
|
||||||
|
#include <3ds.h>
|
||||||
|
#include <citro3d.h>
|
||||||
|
#include <tex3ds.h>
|
||||||
|
|
||||||
|
#include "gfx.h"
|
||||||
|
|
||||||
|
#include <cassert>
|
||||||
|
#include <chrono>
|
||||||
|
#include <ctime>
|
||||||
|
#include <malloc.h>
|
||||||
|
|
||||||
|
namespace
|
||||||
|
{
|
||||||
|
constexpr auto STACK_SIZE = 32768;
|
||||||
|
constexpr auto SOCU_ALIGN = 0x1000;
|
||||||
|
constexpr auto SOCU_BUFFERSIZE = 0x100000;
|
||||||
|
|
||||||
|
static_assert (SOCU_BUFFERSIZE % SOCU_ALIGN == 0);
|
||||||
|
|
||||||
|
bool s_socuActive = false;
|
||||||
|
u32 *s_socuBuffer = nullptr;
|
||||||
|
|
||||||
|
C3D_Tex s_gfxTexture;
|
||||||
|
Tex3DS_Texture s_gfxT3x;
|
||||||
|
|
||||||
|
void startNetwork ()
|
||||||
|
{
|
||||||
|
if (s_socuActive)
|
||||||
|
return;
|
||||||
|
|
||||||
|
std::uint32_t wifi = 0;
|
||||||
|
if (R_FAILED (ACU_GetWifiStatus (&wifi)) || !wifi)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (!s_socuBuffer)
|
||||||
|
s_socuBuffer = static_cast<u32 *> (::memalign (SOCU_ALIGN, SOCU_BUFFERSIZE));
|
||||||
|
|
||||||
|
if (!s_socuBuffer)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (R_FAILED (socInit (s_socuBuffer, SOCU_BUFFERSIZE)))
|
||||||
|
return;
|
||||||
|
|
||||||
|
s_socuActive = true;
|
||||||
|
Log::info ("Wifi connected\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
void drawLogo ()
|
||||||
|
{
|
||||||
|
auto subTex = Tex3DS_GetSubTexture (s_gfxT3x, gfx_c3dlogo_idx);
|
||||||
|
|
||||||
|
ImGuiIO &io = ImGui::GetIO ();
|
||||||
|
auto const screenWidth = io.DisplaySize.x;
|
||||||
|
auto const screenHeight = io.DisplaySize.y;
|
||||||
|
auto const logoWidth = subTex->width / io.DisplayFramebufferScale.x;
|
||||||
|
auto const logoHeight = subTex->height / io.DisplayFramebufferScale.y;
|
||||||
|
|
||||||
|
auto const x1 = (screenWidth - logoWidth) / 2.0f;
|
||||||
|
auto const x2 = x1 + logoWidth;
|
||||||
|
auto const y1 = (screenHeight / 2.0f - logoHeight) / 2.0f;
|
||||||
|
auto const y2 = y1 + logoHeight;
|
||||||
|
|
||||||
|
auto const uv1 = ImVec2 (subTex->left, subTex->top);
|
||||||
|
auto const uv2 = ImVec2 (subTex->right, subTex->bottom);
|
||||||
|
|
||||||
|
ImGui::GetBackgroundDrawList ()->AddImage (
|
||||||
|
&s_gfxTexture, ImVec2 (x1, y1), ImVec2 (x2, y2), uv1, uv2);
|
||||||
|
|
||||||
|
ImGui::GetBackgroundDrawList ()->AddImage (&s_gfxTexture,
|
||||||
|
ImVec2 (x1, y1 + screenHeight / 2.0f),
|
||||||
|
ImVec2 (x2, y2 + screenHeight / 2.0f),
|
||||||
|
uv1,
|
||||||
|
uv2);
|
||||||
|
}
|
||||||
|
|
||||||
|
void drawStatus ()
|
||||||
|
{
|
||||||
|
constexpr unsigned batteryLevels[] = {
|
||||||
|
gfx_battery0_idx,
|
||||||
|
gfx_battery0_idx,
|
||||||
|
gfx_battery1_idx,
|
||||||
|
gfx_battery2_idx,
|
||||||
|
gfx_battery3_idx,
|
||||||
|
gfx_battery4_idx,
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr unsigned wifiLevels[] = {
|
||||||
|
gfx_wifiNull_idx,
|
||||||
|
gfx_wifi1_idx,
|
||||||
|
gfx_wifi2_idx,
|
||||||
|
gfx_wifi3_idx,
|
||||||
|
};
|
||||||
|
|
||||||
|
static u8 charging = 0;
|
||||||
|
static u8 level = 5;
|
||||||
|
PTMU_GetBatteryChargeState (&charging);
|
||||||
|
if (!charging)
|
||||||
|
{
|
||||||
|
PTMU_GetBatteryLevel (&level);
|
||||||
|
if (level >= std::extent_v<decltype (batteryLevels)>)
|
||||||
|
svcBreak (USERBREAK_PANIC);
|
||||||
|
}
|
||||||
|
|
||||||
|
auto const &io = ImGui::GetIO ();
|
||||||
|
auto const &style = ImGui::GetStyle ();
|
||||||
|
|
||||||
|
auto const screenWidth = io.DisplaySize.x;
|
||||||
|
|
||||||
|
auto const battery =
|
||||||
|
Tex3DS_GetSubTexture (s_gfxT3x, charging ? gfx_batteryCharge_idx : batteryLevels[level]);
|
||||||
|
auto const batteryWidth = battery->width / io.DisplayFramebufferScale.x;
|
||||||
|
auto const batteryHeight = battery->height / io.DisplayFramebufferScale.y;
|
||||||
|
|
||||||
|
auto const p1 = ImVec2 (screenWidth - batteryWidth, 0.0f);
|
||||||
|
auto const p2 = ImVec2 (screenWidth, batteryHeight);
|
||||||
|
|
||||||
|
auto const uv1 = ImVec2 (battery->left, battery->top);
|
||||||
|
auto const uv2 = ImVec2 (battery->right, battery->bottom);
|
||||||
|
|
||||||
|
ImGui::GetForegroundDrawList ()->AddImage (&s_gfxTexture, p1, p2, uv1, uv2);
|
||||||
|
|
||||||
|
auto const wifiStrength = osGetWifiStrength ();
|
||||||
|
|
||||||
|
auto const wifi = Tex3DS_GetSubTexture (s_gfxT3x, wifiLevels[wifiStrength]);
|
||||||
|
auto const wifiWidth = wifi->width / io.DisplayFramebufferScale.x;
|
||||||
|
auto const wifiHeight = wifi->height / io.DisplayFramebufferScale.y;
|
||||||
|
|
||||||
|
auto const p3 = ImVec2 (p1.x - wifiWidth - 4.0f, 0.0f);
|
||||||
|
auto const p4 = ImVec2 (p1.x - 4.0f, wifiHeight);
|
||||||
|
|
||||||
|
auto const uv3 = ImVec2 (wifi->left, wifi->top);
|
||||||
|
auto const uv4 = ImVec2 (wifi->right, wifi->bottom);
|
||||||
|
|
||||||
|
ImGui::GetForegroundDrawList ()->AddImage (&s_gfxTexture, p3, p4, uv3, uv4);
|
||||||
|
|
||||||
|
char buffer[64];
|
||||||
|
auto const now = std::time (nullptr);
|
||||||
|
std::strftime (buffer, sizeof (buffer), "%H:%M:%S", std::localtime (&now));
|
||||||
|
ImGui::GetForegroundDrawList ()->AddText (
|
||||||
|
ImVec2 (p3.x - 130.0f, style.FramePadding.y), 0xFFFFFFFF, buffer);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool platform::init ()
|
||||||
|
{
|
||||||
|
osSetSpeedupEnable (true);
|
||||||
|
|
||||||
|
acInit ();
|
||||||
|
ptmuInit ();
|
||||||
|
romfsInit ();
|
||||||
|
gfxInitDefault ();
|
||||||
|
gfxSet3D (false);
|
||||||
|
sdmcWriteSafe (false);
|
||||||
|
|
||||||
|
#ifndef NDEBUG
|
||||||
|
consoleDebugInit (debugDevice_SVC);
|
||||||
|
std::setvbuf (stderr, nullptr, _IOLBF, 0);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
IMGUI_CHECKVERSION ();
|
||||||
|
ImGui::CreateContext ();
|
||||||
|
|
||||||
|
if (!imgui::ctru::init ())
|
||||||
|
{
|
||||||
|
ImGui::DestroyContext ();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
imgui::citro3d::init ();
|
||||||
|
|
||||||
|
{
|
||||||
|
fs::File file;
|
||||||
|
if (!file.open ("romfs:/gfx.t3x"))
|
||||||
|
svcBreak (USERBREAK_PANIC);
|
||||||
|
|
||||||
|
s_gfxT3x = Tex3DS_TextureImportStdio (file, &s_gfxTexture, nullptr, true);
|
||||||
|
if (!s_gfxT3x)
|
||||||
|
svcBreak (USERBREAK_PANIC);
|
||||||
|
}
|
||||||
|
|
||||||
|
C3D_TexSetFilter (&s_gfxTexture, GPU_LINEAR, GPU_LINEAR);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool platform::loop ()
|
||||||
|
{
|
||||||
|
if (!aptMainLoop ())
|
||||||
|
return false;
|
||||||
|
|
||||||
|
startNetwork ();
|
||||||
|
|
||||||
|
hidScanInput ();
|
||||||
|
|
||||||
|
if (hidKeysDown () & KEY_START)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
imgui::ctru::newFrame ();
|
||||||
|
imgui::citro3d::newFrame ();
|
||||||
|
ImGui::NewFrame ();
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void platform::render ()
|
||||||
|
{
|
||||||
|
drawLogo ();
|
||||||
|
drawStatus ();
|
||||||
|
|
||||||
|
ImGui::Render ();
|
||||||
|
|
||||||
|
imgui::citro3d::render ();
|
||||||
|
}
|
||||||
|
|
||||||
|
void platform::exit ()
|
||||||
|
{
|
||||||
|
Tex3DS_TextureFree (s_gfxT3x);
|
||||||
|
C3D_TexDelete (&s_gfxTexture);
|
||||||
|
|
||||||
|
ImGui::DestroyContext ();
|
||||||
|
|
||||||
|
imgui::citro3d::exit ();
|
||||||
|
imgui::ctru::exit ();
|
||||||
|
|
||||||
|
socExit ();
|
||||||
|
s_socuActive = false;
|
||||||
|
std::free (s_socuBuffer);
|
||||||
|
|
||||||
|
gfxExit ();
|
||||||
|
ptmuExit ();
|
||||||
|
acExit ();
|
||||||
|
}
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////
|
||||||
|
class platform::Thread::privateData_t
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
privateData_t ()
|
||||||
|
{
|
||||||
|
if (thread)
|
||||||
|
threadFree (thread);
|
||||||
|
}
|
||||||
|
|
||||||
|
privateData_t (std::function<void ()> func_) : thread (nullptr)
|
||||||
|
{
|
||||||
|
s32 priority = 0x30;
|
||||||
|
svcGetThreadPriority (&priority, CUR_THREAD_HANDLE);
|
||||||
|
|
||||||
|
thread = threadCreate (&privateData_t::threadFunc, this, STACK_SIZE, priority, -1, false);
|
||||||
|
assert (thread);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void threadFunc (void *const arg_)
|
||||||
|
{
|
||||||
|
auto const t = static_cast<privateData_t *> (arg_);
|
||||||
|
t->func ();
|
||||||
|
}
|
||||||
|
|
||||||
|
::Thread thread = nullptr;
|
||||||
|
std::function<void ()> func;
|
||||||
|
};
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////
|
||||||
|
platform::Thread::~Thread () = default;
|
||||||
|
|
||||||
|
platform::Thread::Thread () : m_d (new privateData_t ())
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
platform::Thread::Thread (std::function<void ()> func_) : m_d (new privateData_t (func_))
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
platform::Thread::Thread (Thread &&that_) : m_d (new privateData_t ())
|
||||||
|
{
|
||||||
|
std::swap (m_d, that_.m_d);
|
||||||
|
}
|
||||||
|
|
||||||
|
platform::Thread &platform::Thread::operator= (Thread &&that_)
|
||||||
|
{
|
||||||
|
std::swap (m_d, that_.m_d);
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
|
void platform::Thread::join ()
|
||||||
|
{
|
||||||
|
threadJoin (m_d->thread, UINT64_MAX);
|
||||||
|
}
|
||||||
|
|
||||||
|
void platform::Thread::sleep (std::chrono::milliseconds const timeout_)
|
||||||
|
{
|
||||||
|
svcSleepThread (std::chrono::nanoseconds (timeout_).count ());
|
||||||
|
}
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////
|
||||||
|
class platform::Mutex::privateData_t
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
LightLock mutex;
|
||||||
|
};
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////
|
||||||
|
platform::Mutex::~Mutex () = default;
|
||||||
|
|
||||||
|
platform::Mutex::Mutex () : m_d (new privateData_t ())
|
||||||
|
{
|
||||||
|
LightLock_Init (&m_d->mutex);
|
||||||
|
}
|
||||||
|
|
||||||
|
void platform::Mutex::lock ()
|
||||||
|
{
|
||||||
|
LightLock_Lock (&m_d->mutex);
|
||||||
|
}
|
||||||
|
|
||||||
|
void platform::Mutex::unlock ()
|
||||||
|
{
|
||||||
|
LightLock_Unlock (&m_d->mutex);
|
||||||
|
}
|
61
source/3ds/vshader.v.pica
Normal file
@ -0,0 +1,61 @@
|
|||||||
|
; ftpd is a server implementation based on the following:
|
||||||
|
; - RFC 959 (https://tools.ietf.org/html/rfc959)
|
||||||
|
; - RFC 3659 (https://tools.ietf.org/html/rfc3659)
|
||||||
|
; - suggested implementation details from https://cr.yp.to/ftp/filesystem.html
|
||||||
|
;
|
||||||
|
; Copyright (C) 2020 Michael Theall
|
||||||
|
;
|
||||||
|
; 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 <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
; Example PICA200 vertex shader
|
||||||
|
|
||||||
|
; Uniforms
|
||||||
|
.fvec proj[4]
|
||||||
|
|
||||||
|
; Constants
|
||||||
|
.constf constants(1.0, 0.0, 0.00392156862745, 0.0)
|
||||||
|
|
||||||
|
; Outputs
|
||||||
|
.out outPos position
|
||||||
|
.out outUv texcoord0
|
||||||
|
.out outColor color
|
||||||
|
|
||||||
|
; Inputs (defined as aliases for convenience)
|
||||||
|
.alias inPos v0
|
||||||
|
.alias inUv v1
|
||||||
|
.alias inColor v2
|
||||||
|
|
||||||
|
.proc main
|
||||||
|
; Force inPos.z = 0.0, inPos.w = 1.0
|
||||||
|
mov r0.xy, inPos.xy
|
||||||
|
mov r0.zw, constants.yx
|
||||||
|
|
||||||
|
; outPos = proj * inPos
|
||||||
|
dp4 outPos.x, proj[0], r0
|
||||||
|
dp4 outPos.y, proj[1], r0
|
||||||
|
dp4 outPos.z, proj[2], r0
|
||||||
|
dp4 outPos.w, proj[3], r0
|
||||||
|
|
||||||
|
; outUv = inUv
|
||||||
|
mov outUv, inUv
|
||||||
|
|
||||||
|
; normalize inColor
|
||||||
|
mul r1, constants.zzzz, inColor
|
||||||
|
|
||||||
|
; outColor = inColor
|
||||||
|
mov outColor, r1
|
||||||
|
|
||||||
|
; We're finished
|
||||||
|
end
|
||||||
|
.end
|
276
source/console.c
@ -1,276 +0,0 @@
|
|||||||
#include "console.h"
|
|
||||||
#include <arpa/inet.h>
|
|
||||||
#include <errno.h>
|
|
||||||
#include <stdarg.h>
|
|
||||||
#include <stdio.h>
|
|
||||||
#include <stdlib.h>
|
|
||||||
#include <string.h>
|
|
||||||
|
|
||||||
#ifdef _3DS
|
|
||||||
#include <3ds.h>
|
|
||||||
#define CONSOLE_WIDTH 50
|
|
||||||
#define CONSOLE_HEIGHT 30
|
|
||||||
#elif defined(__SWITCH__)
|
|
||||||
#include <switch.h>
|
|
||||||
#define CONSOLE_WIDTH 80
|
|
||||||
#define CONSOLE_HEIGHT 45
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if defined(_3DS) || defined (__SWITCH__)
|
|
||||||
static PrintConsole status_console;
|
|
||||||
static PrintConsole main_console;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if ENABLE_LOGGING
|
|
||||||
static bool disable_logging = false;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if defined(_3DS)
|
|
||||||
static PrintConsole tcp_console;
|
|
||||||
|
|
||||||
/*! initialize console subsystem */
|
|
||||||
void
|
|
||||||
console_init(void)
|
|
||||||
{
|
|
||||||
consoleInit(GFX_TOP, &status_console);
|
|
||||||
consoleSetWindow(&status_console, 0, 0, 50, 1);
|
|
||||||
|
|
||||||
consoleInit(GFX_TOP, &main_console);
|
|
||||||
consoleSetWindow(&main_console, 0, 1, 50, 29);
|
|
||||||
|
|
||||||
consoleInit(GFX_BOTTOM, &tcp_console);
|
|
||||||
|
|
||||||
consoleSelect(&main_console);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/*! print tcp tables */
|
|
||||||
static void
|
|
||||||
print_tcp_table(void)
|
|
||||||
{
|
|
||||||
static SOCU_TCPTableEntry tcp_entries[32];
|
|
||||||
socklen_t optlen;
|
|
||||||
size_t i;
|
|
||||||
int rc, lines = 0;
|
|
||||||
|
|
||||||
#ifdef ENABLE_LOGGING
|
|
||||||
disable_logging = true;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
consoleSelect(&tcp_console);
|
|
||||||
console_print("\x1b[0;0H\x1b[K");
|
|
||||||
optlen = sizeof(tcp_entries);
|
|
||||||
rc = SOCU_GetNetworkOpt(SOL_CONFIG, NETOPT_TCP_TABLE, tcp_entries, &optlen);
|
|
||||||
if(rc != 0 && errno != ENODEV)
|
|
||||||
console_print(RED "tcp table: %d %s\n\x1b[J\n" RESET, errno, strerror(errno));
|
|
||||||
else if(rc == 0)
|
|
||||||
{
|
|
||||||
for(i = 0; lines < 30 && i < optlen / sizeof(SOCU_TCPTableEntry); ++i)
|
|
||||||
{
|
|
||||||
SOCU_TCPTableEntry *entry = &tcp_entries[i];
|
|
||||||
struct sockaddr_in *local = (struct sockaddr_in*)&entry->local;
|
|
||||||
struct sockaddr_in *remote = (struct sockaddr_in*)&entry->remote;
|
|
||||||
|
|
||||||
console_print(GREEN "%stcp[%zu]: ", i == 0 ? "" : "\n", i);
|
|
||||||
switch(entry->state)
|
|
||||||
{
|
|
||||||
case TCP_STATE_CLOSED:
|
|
||||||
console_print("CLOSED\x1b[K");
|
|
||||||
local = remote = NULL;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case TCP_STATE_LISTEN:
|
|
||||||
console_print("LISTEN\x1b[K");
|
|
||||||
remote = NULL;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case TCP_STATE_ESTABLISHED:
|
|
||||||
console_print("ESTABLISHED\x1b[K");
|
|
||||||
break;
|
|
||||||
|
|
||||||
case TCP_STATE_FINWAIT1:
|
|
||||||
console_print("FINWAIT1\x1b[K");
|
|
||||||
break;
|
|
||||||
|
|
||||||
case TCP_STATE_FINWAIT2:
|
|
||||||
console_print("FINWAIT2\x1b[K");
|
|
||||||
break;
|
|
||||||
|
|
||||||
case TCP_STATE_CLOSE_WAIT:
|
|
||||||
console_print("CLOSE_WAIT\x1b[K");
|
|
||||||
break;
|
|
||||||
|
|
||||||
case TCP_STATE_LAST_ACK:
|
|
||||||
console_print("LAST_ACK\x1b[K");
|
|
||||||
break;
|
|
||||||
|
|
||||||
case TCP_STATE_TIME_WAIT:
|
|
||||||
console_print("TIME_WAIT\x1b[K");
|
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
|
||||||
console_print("State %lu\x1b[K", entry->state);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
++lines;
|
|
||||||
|
|
||||||
if(local && (lines++ < 30))
|
|
||||||
console_print("\n Local %s:%u\x1b[K", inet_ntoa(local->sin_addr),
|
|
||||||
ntohs(local->sin_port));
|
|
||||||
|
|
||||||
if(remote && (lines++ < 30))
|
|
||||||
console_print("\n Peer %s:%u\x1b[K", inet_ntoa(remote->sin_addr),
|
|
||||||
ntohs(remote->sin_port));
|
|
||||||
}
|
|
||||||
|
|
||||||
console_print(RESET "\x1b[J");
|
|
||||||
}
|
|
||||||
else
|
|
||||||
console_print("\x1b[2J");
|
|
||||||
|
|
||||||
consoleSelect(&main_console);
|
|
||||||
|
|
||||||
#ifdef ENABLE_LOGGING
|
|
||||||
disable_logging = false;
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
#elif defined(__SWITCH__)
|
|
||||||
/*! initialize console subsystem */
|
|
||||||
void
|
|
||||||
console_init(void)
|
|
||||||
{
|
|
||||||
consoleInit(&status_console);
|
|
||||||
consoleSetWindow(&status_console, 0, 0, CONSOLE_WIDTH, 1);
|
|
||||||
|
|
||||||
consoleInit( &main_console);
|
|
||||||
consoleSetWindow(&main_console, 0, 1, CONSOLE_WIDTH, CONSOLE_HEIGHT-1);
|
|
||||||
|
|
||||||
consoleSelect(&main_console);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if defined(_3DS) || defined(__SWITCH__)
|
|
||||||
|
|
||||||
|
|
||||||
/*! set status bar contents
|
|
||||||
*
|
|
||||||
* @param[in] fmt format string
|
|
||||||
* @param[in] ... format arguments
|
|
||||||
*/
|
|
||||||
void
|
|
||||||
console_set_status(const char *fmt, ...)
|
|
||||||
{
|
|
||||||
va_list ap;
|
|
||||||
|
|
||||||
consoleSelect(&status_console);
|
|
||||||
va_start(ap, fmt);
|
|
||||||
vprintf(fmt, ap);
|
|
||||||
#ifdef ENABLE_LOGGING
|
|
||||||
vfprintf(stderr, fmt, ap);
|
|
||||||
#endif
|
|
||||||
va_end(ap);
|
|
||||||
consoleSelect(&main_console);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*! add text to the console
|
|
||||||
*
|
|
||||||
* @param[in] fmt format string
|
|
||||||
* @param[in] ... format arguments
|
|
||||||
*/
|
|
||||||
void
|
|
||||||
console_print(const char *fmt, ...)
|
|
||||||
{
|
|
||||||
va_list ap;
|
|
||||||
|
|
||||||
va_start(ap, fmt);
|
|
||||||
vprintf(fmt, ap);
|
|
||||||
#ifdef ENABLE_LOGGING
|
|
||||||
if(!disable_logging)
|
|
||||||
vfprintf(stderr, fmt, ap);
|
|
||||||
#endif
|
|
||||||
va_end(ap);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*! print debug message
|
|
||||||
*
|
|
||||||
* @param[in] fmt format string
|
|
||||||
* @param[in] ... format arguments
|
|
||||||
*/
|
|
||||||
void
|
|
||||||
debug_print(const char *fmt, ...)
|
|
||||||
{
|
|
||||||
#ifdef ENABLE_LOGGING
|
|
||||||
va_list ap;
|
|
||||||
|
|
||||||
va_start(ap, fmt);
|
|
||||||
vfprintf(stderr, fmt, ap);
|
|
||||||
va_end(ap);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
/*! draw console to screen */
|
|
||||||
void
|
|
||||||
console_render(void)
|
|
||||||
{
|
|
||||||
/* print tcp table */
|
|
||||||
#ifdef _3DS
|
|
||||||
print_tcp_table();
|
|
||||||
#endif
|
|
||||||
/* flush framebuffer */
|
|
||||||
#ifdef _3DS
|
|
||||||
gfxFlushBuffers();
|
|
||||||
gspWaitForVBlank();
|
|
||||||
gfxSwapBuffers();
|
|
||||||
#endif
|
|
||||||
#ifdef __SWITCH__
|
|
||||||
consoleUpdate(NULL);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
#else
|
|
||||||
|
|
||||||
/* this is a lot easier when you have a real console */
|
|
||||||
|
|
||||||
void
|
|
||||||
console_init(void)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
console_set_status(const char *fmt, ...)
|
|
||||||
{
|
|
||||||
va_list ap;
|
|
||||||
va_start(ap, fmt);
|
|
||||||
vprintf(fmt, ap);
|
|
||||||
va_end(ap);
|
|
||||||
fputc('\n', stdout);
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
console_print(const char *fmt, ...)
|
|
||||||
{
|
|
||||||
va_list ap;
|
|
||||||
va_start(ap, fmt);
|
|
||||||
vprintf(fmt, ap);
|
|
||||||
va_end(ap);
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
debug_print(const char *fmt, ...)
|
|
||||||
{
|
|
||||||
#ifdef ENABLE_LOGGING
|
|
||||||
va_list ap;
|
|
||||||
va_start(ap, fmt);
|
|
||||||
vfprintf(stderr, fmt, ap);
|
|
||||||
va_end(ap);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
void console_render(void)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
211
source/fs.cpp
Normal file
@ -0,0 +1,211 @@
|
|||||||
|
// ftpd is a server implementation based on the following:
|
||||||
|
// - RFC 959 (https://tools.ietf.org/html/rfc959)
|
||||||
|
// - RFC 3659 (https://tools.ietf.org/html/rfc3659)
|
||||||
|
// - suggested implementation details from https://cr.yp.to/ftp/filesystem.html
|
||||||
|
//
|
||||||
|
// Copyright (C) 2020 Michael Theall
|
||||||
|
//
|
||||||
|
// 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 <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
#include "fs.h"
|
||||||
|
|
||||||
|
#include <cinttypes>
|
||||||
|
#include <cstdio>
|
||||||
|
|
||||||
|
std::string fs::printSize (std::uint64_t const size_)
|
||||||
|
{
|
||||||
|
constexpr std::uint64_t const KiB = 1024;
|
||||||
|
constexpr std::uint64_t const MiB = 1024 * KiB;
|
||||||
|
constexpr std::uint64_t const GiB = 1024 * MiB;
|
||||||
|
constexpr std::uint64_t const TiB = 1024 * GiB;
|
||||||
|
constexpr std::uint64_t const PiB = 1024 * TiB;
|
||||||
|
constexpr std::uint64_t const EiB = 1024 * PiB;
|
||||||
|
|
||||||
|
char buffer[64] = {};
|
||||||
|
|
||||||
|
for (auto const &[name, bin] : {
|
||||||
|
// clang-format off
|
||||||
|
std::make_pair ("EiB", EiB),
|
||||||
|
std::make_pair ("PiB", PiB),
|
||||||
|
std::make_pair ("TiB", TiB),
|
||||||
|
std::make_pair ("GiB", GiB),
|
||||||
|
std::make_pair ("MiB", MiB),
|
||||||
|
std::make_pair ("KiB", KiB)
|
||||||
|
// clang-format on
|
||||||
|
})
|
||||||
|
{
|
||||||
|
auto const whole = size_ / bin;
|
||||||
|
if (size_ >= 100 * bin)
|
||||||
|
{
|
||||||
|
std::sprintf (buffer, "%" PRIu64 "%s", whole, name);
|
||||||
|
return buffer;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto const frac = size_ - whole * bin;
|
||||||
|
if (size_ >= 10 * bin)
|
||||||
|
{
|
||||||
|
std::sprintf (buffer, "%" PRIu64 ".%" PRIu64 "%s", whole, frac * 10 / bin, name);
|
||||||
|
return buffer;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (size_ >= 1000 * (bin / KiB))
|
||||||
|
{
|
||||||
|
std::sprintf (buffer, "%" PRIu64 ".%02" PRIu64 "%s", whole, frac * 100 / bin, name);
|
||||||
|
return buffer;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
std::sprintf (buffer, "%" PRIu64, size_);
|
||||||
|
return buffer;
|
||||||
|
}
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////
|
||||||
|
fs::File::~File () = default;
|
||||||
|
|
||||||
|
fs::File::File () = default;
|
||||||
|
|
||||||
|
fs::File::File (File &&that_) = default;
|
||||||
|
|
||||||
|
fs::File &fs::File::operator= (File &&that_) = default;
|
||||||
|
|
||||||
|
fs::File::operator bool () const
|
||||||
|
{
|
||||||
|
return static_cast<bool> (m_fp);
|
||||||
|
}
|
||||||
|
|
||||||
|
fs::File::operator FILE * () const
|
||||||
|
{
|
||||||
|
return m_fp.get ();
|
||||||
|
}
|
||||||
|
|
||||||
|
void fs::File::setBufferSize (std::size_t const size_)
|
||||||
|
{
|
||||||
|
if (m_bufferSize != size_)
|
||||||
|
{
|
||||||
|
m_buffer = std::make_unique<char[]> (size_);
|
||||||
|
m_bufferSize = size_;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (m_fp)
|
||||||
|
std::setvbuf (m_fp.get (), m_buffer.get (), _IOFBF, m_bufferSize);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool fs::File::open (char const *const path_, char const *const mode_)
|
||||||
|
{
|
||||||
|
auto const fp = std::fopen (path_, mode_);
|
||||||
|
if (!fp)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
m_fp = std::unique_ptr<std::FILE, int (*) (std::FILE *)> (fp, &std::fclose);
|
||||||
|
|
||||||
|
if (m_buffer)
|
||||||
|
std::setvbuf (m_fp.get (), m_buffer.get (), _IOFBF, m_bufferSize);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void fs::File::close ()
|
||||||
|
{
|
||||||
|
m_fp.reset ();
|
||||||
|
}
|
||||||
|
|
||||||
|
ssize_t fs::File::seek (std::size_t const pos_, int const origin_)
|
||||||
|
{
|
||||||
|
return std::fseek (m_fp.get (), pos_, origin_);
|
||||||
|
}
|
||||||
|
|
||||||
|
ssize_t fs::File::read (void *const data_, std::size_t const size_)
|
||||||
|
{
|
||||||
|
return std::fread (data_, 1, size_, m_fp.get ());
|
||||||
|
}
|
||||||
|
|
||||||
|
bool fs::File::readAll (void *const data_, std::size_t const size_)
|
||||||
|
{
|
||||||
|
auto p = static_cast<char *> (data_);
|
||||||
|
std::size_t bytes = 0;
|
||||||
|
|
||||||
|
while (bytes < size_)
|
||||||
|
{
|
||||||
|
auto const rc = read (p, size_ - bytes);
|
||||||
|
if (rc <= 0)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
p += rc;
|
||||||
|
bytes += rc;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
ssize_t fs::File::write (void const *const data_, std::size_t const size_)
|
||||||
|
{
|
||||||
|
return std::fwrite (data_, 1, size_, m_fp.get ());
|
||||||
|
}
|
||||||
|
|
||||||
|
bool fs::File::writeAll (void const *const data_, std::size_t const size_)
|
||||||
|
{
|
||||||
|
auto p = static_cast<char const *> (data_);
|
||||||
|
std::size_t bytes = 0;
|
||||||
|
|
||||||
|
while (bytes < size_)
|
||||||
|
{
|
||||||
|
auto const rc = write (p, size_ - bytes);
|
||||||
|
if (rc <= 0)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
p += rc;
|
||||||
|
bytes += rc;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////
|
||||||
|
fs::Dir::~Dir () = default;
|
||||||
|
|
||||||
|
fs::Dir::Dir () = default;
|
||||||
|
|
||||||
|
fs::Dir::Dir (Dir &&that_) = default;
|
||||||
|
|
||||||
|
fs::Dir &fs::Dir::operator= (Dir &&that_) = default;
|
||||||
|
|
||||||
|
fs::Dir::operator bool () const
|
||||||
|
{
|
||||||
|
return static_cast<bool> (m_dp);
|
||||||
|
}
|
||||||
|
|
||||||
|
fs::Dir::operator DIR * () const
|
||||||
|
{
|
||||||
|
return m_dp.get ();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool fs::Dir::open (char const *const path_)
|
||||||
|
{
|
||||||
|
auto const dp = ::opendir (path_);
|
||||||
|
if (!dp)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
m_dp = std::unique_ptr<DIR, int (*) (DIR *)> (dp, &::closedir);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void fs::Dir::close ()
|
||||||
|
{
|
||||||
|
m_dp.reset ();
|
||||||
|
}
|
||||||
|
|
||||||
|
struct dirent *fs::Dir::read ()
|
||||||
|
{
|
||||||
|
return ::readdir (m_dp.get ());
|
||||||
|
}
|
4114
source/ftp.c
253
source/ftpServer.cpp
Normal file
@ -0,0 +1,253 @@
|
|||||||
|
// ftpd is a server implementation based on the following:
|
||||||
|
// - RFC 959 (https://tools.ietf.org/html/rfc959)
|
||||||
|
// - RFC 3659 (https://tools.ietf.org/html/rfc3659)
|
||||||
|
// - suggested implementation details from https://cr.yp.to/ftp/filesystem.html
|
||||||
|
//
|
||||||
|
// Copyright (C) 2020 Michael Theall
|
||||||
|
//
|
||||||
|
// 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 <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
#include "ftpServer.h"
|
||||||
|
|
||||||
|
#include "fs.h"
|
||||||
|
|
||||||
|
#include "imgui.h"
|
||||||
|
|
||||||
|
#include <arpa/inet.h>
|
||||||
|
#include <poll.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
|
||||||
|
#include <chrono>
|
||||||
|
#include <cstdio>
|
||||||
|
#include <cstring>
|
||||||
|
#include <mutex>
|
||||||
|
#include <thread>
|
||||||
|
using namespace std::chrono_literals;
|
||||||
|
|
||||||
|
#ifndef _3DS
|
||||||
|
#define MULTITHREADED 1
|
||||||
|
#else
|
||||||
|
#define MULTITHREADED 0
|
||||||
|
#endif
|
||||||
|
|
||||||
|
namespace
|
||||||
|
{
|
||||||
|
auto const s_startTime = std::chrono::system_clock::to_time_t (std::chrono::system_clock::now ());
|
||||||
|
platform::Mutex s_lock;
|
||||||
|
std::string s_freeSpace;
|
||||||
|
}
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////
|
||||||
|
FtpServer::~FtpServer ()
|
||||||
|
{
|
||||||
|
m_quit = true;
|
||||||
|
|
||||||
|
m_thread.join ();
|
||||||
|
}
|
||||||
|
|
||||||
|
FtpServer::FtpServer (std::uint16_t const port_)
|
||||||
|
: m_log (Log::create ()), m_port (port_), m_quit (false)
|
||||||
|
{
|
||||||
|
Log::bind (m_log);
|
||||||
|
|
||||||
|
handleStartButton ();
|
||||||
|
|
||||||
|
#if MULTITHREADED
|
||||||
|
m_thread = platform::Thread (std::bind (&FtpServer::threadFunc, this));
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
void FtpServer::draw ()
|
||||||
|
{
|
||||||
|
#if !MULTITHREADED
|
||||||
|
loop ();
|
||||||
|
#endif
|
||||||
|
|
||||||
|
ImGuiIO &io = ImGui::GetIO ();
|
||||||
|
auto const width = io.DisplaySize.x;
|
||||||
|
auto const height = io.DisplaySize.y;
|
||||||
|
|
||||||
|
ImGui::SetNextWindowPos (ImVec2 (0, 0), ImGuiCond_FirstUseEver);
|
||||||
|
#ifdef _3DS
|
||||||
|
// top screen
|
||||||
|
ImGui::SetNextWindowSize (ImVec2 (width, height / 2.0f));
|
||||||
|
#else
|
||||||
|
ImGui::SetNextWindowSize (ImVec2 (width, height));
|
||||||
|
#endif
|
||||||
|
ImGui::Begin (STATUS_STRING,
|
||||||
|
nullptr,
|
||||||
|
ImGuiWindowFlags_NoCollapse | ImGuiWindowFlags_NoMove | ImGuiWindowFlags_NoResize);
|
||||||
|
|
||||||
|
{
|
||||||
|
auto const lock = std::scoped_lock (m_lock);
|
||||||
|
if (!m_socket)
|
||||||
|
{
|
||||||
|
if (ImGui::Button ("Start"))
|
||||||
|
handleStartButton ();
|
||||||
|
}
|
||||||
|
else if (ImGui::Button ("Stop"))
|
||||||
|
handleStopButton ();
|
||||||
|
|
||||||
|
if (m_socket)
|
||||||
|
{
|
||||||
|
ImGui::SameLine ();
|
||||||
|
ImGui::TextUnformatted (m_name.c_str ());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
auto const lock = std::scoped_lock (s_lock);
|
||||||
|
if (!s_freeSpace.empty ())
|
||||||
|
{
|
||||||
|
ImGui::SameLine ();
|
||||||
|
ImGui::TextUnformatted (s_freeSpace.c_str ());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ImGui::Separator ();
|
||||||
|
|
||||||
|
#ifdef _3DS
|
||||||
|
ImGui::BeginChild ("Logs", ImVec2 (0, 0), false, ImGuiWindowFlags_HorizontalScrollbar);
|
||||||
|
#else
|
||||||
|
ImGui::BeginChild ("Logs", ImVec2 (0, 200), false, ImGuiWindowFlags_HorizontalScrollbar);
|
||||||
|
#endif
|
||||||
|
m_log->draw ();
|
||||||
|
ImGui::EndChild ();
|
||||||
|
|
||||||
|
#ifdef _3DS
|
||||||
|
ImGui::End ();
|
||||||
|
|
||||||
|
// bottom screen
|
||||||
|
ImGui::SetNextWindowSize (ImVec2 (width * 0.8f, height / 2.0f));
|
||||||
|
ImGui::SetNextWindowPos (ImVec2 (width * 0.1f, height / 2.0f), ImGuiCond_FirstUseEver);
|
||||||
|
ImGui::Begin ("Sessions",
|
||||||
|
nullptr,
|
||||||
|
ImGuiWindowFlags_NoCollapse | ImGuiWindowFlags_NoMove | ImGuiWindowFlags_NoResize);
|
||||||
|
#else
|
||||||
|
ImGui::Separator ();
|
||||||
|
#endif
|
||||||
|
|
||||||
|
for (auto &session : m_sessions)
|
||||||
|
session->draw ();
|
||||||
|
|
||||||
|
ImGui::End ();
|
||||||
|
}
|
||||||
|
|
||||||
|
UniqueFtpServer FtpServer::create (std::uint16_t const port_)
|
||||||
|
{
|
||||||
|
updateFreeSpace ();
|
||||||
|
return UniqueFtpServer (new FtpServer (port_));
|
||||||
|
}
|
||||||
|
|
||||||
|
void FtpServer::updateFreeSpace ()
|
||||||
|
{
|
||||||
|
#if defined(_3DS) || defined(__SWITCH__)
|
||||||
|
struct statvfs st;
|
||||||
|
if (::statvfs ("sdmc:/", &st) != 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
auto const lock = std::scoped_lock (s_lock);
|
||||||
|
s_freeSpace = fs::printSize (static_cast<std::uint64_t> (st.f_bsize) * st.f_bfree);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
std::time_t FtpServer::startTime ()
|
||||||
|
{
|
||||||
|
return s_startTime;
|
||||||
|
}
|
||||||
|
|
||||||
|
void FtpServer::handleStartButton ()
|
||||||
|
{
|
||||||
|
if (m_socket)
|
||||||
|
return;
|
||||||
|
|
||||||
|
struct sockaddr_in addr;
|
||||||
|
addr.sin_family = AF_INET;
|
||||||
|
#if defined(_3DS) || defined(__SWITCH__)
|
||||||
|
addr.sin_addr.s_addr = gethostid ();
|
||||||
|
#else
|
||||||
|
addr.sin_addr.s_addr = INADDR_ANY;
|
||||||
|
#endif
|
||||||
|
addr.sin_port = htons (m_port);
|
||||||
|
|
||||||
|
auto socket = Socket::create ();
|
||||||
|
if (!socket)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (m_port != 0 && !socket->setReuseAddress (true))
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (!socket->bind (addr))
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (!socket->listen (10))
|
||||||
|
return;
|
||||||
|
|
||||||
|
auto const &sockName = socket->sockName ();
|
||||||
|
auto const name = sockName.name ();
|
||||||
|
|
||||||
|
m_name.resize (std::strlen (name) + 3 + 5);
|
||||||
|
m_name.resize (std::sprintf (&m_name[0], "[%s]:%u", name, sockName.port ()));
|
||||||
|
|
||||||
|
Log::info ("Started server at %s\n", m_name.c_str ());
|
||||||
|
|
||||||
|
m_socket = std::move (socket);
|
||||||
|
}
|
||||||
|
|
||||||
|
void FtpServer::handleStopButton ()
|
||||||
|
{
|
||||||
|
m_socket.reset ();
|
||||||
|
Log::info ("Stopped server at %s\n", m_name.c_str ());
|
||||||
|
}
|
||||||
|
|
||||||
|
void FtpServer::loop ()
|
||||||
|
{
|
||||||
|
{
|
||||||
|
auto const lock = std::scoped_lock (m_lock);
|
||||||
|
if (m_socket)
|
||||||
|
{
|
||||||
|
Socket::PollInfo info{*m_socket, POLLIN, 0};
|
||||||
|
if (Socket::poll (&info, 1, 0ms) > 0)
|
||||||
|
{
|
||||||
|
auto socket = m_socket->accept ();
|
||||||
|
if (socket)
|
||||||
|
m_sessions.emplace_back (FtpSession::create (std::move (socket)));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (auto it = std::begin (m_sessions); it != std::end (m_sessions);)
|
||||||
|
{
|
||||||
|
auto const &session = *it;
|
||||||
|
if (session->dead ())
|
||||||
|
it = m_sessions.erase (it);
|
||||||
|
else
|
||||||
|
++it;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!m_sessions.empty ())
|
||||||
|
FtpSession::poll (m_sessions);
|
||||||
|
#if MULTITHREADED
|
||||||
|
else
|
||||||
|
platform::Thread::sleep (16ms);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
void FtpServer::threadFunc ()
|
||||||
|
{
|
||||||
|
Log::bind (m_log);
|
||||||
|
|
||||||
|
while (!m_quit)
|
||||||
|
loop ();
|
||||||
|
}
|
2482
source/ftpSession.cpp
Normal file
1
source/imgui/imgui.cpp
Normal file
@ -0,0 +1 @@
|
|||||||
|
#error "Please use https://github.com/ocornut/imgui/releases/tag/v1.75"
|
1
source/imgui/imgui_draw.cpp
Normal file
@ -0,0 +1 @@
|
|||||||
|
#error "Please use https://github.com/ocornut/imgui/releases/tag/v1.75"
|
1
source/imgui/imgui_internal.h
Normal file
@ -0,0 +1 @@
|
|||||||
|
#error "Please use https://github.com/ocornut/imgui/releases/tag/v1.75"
|
1
source/imgui/imgui_widgets.cpp
Normal file
@ -0,0 +1 @@
|
|||||||
|
#error "Please use https://github.com/ocornut/imgui/releases/tag/v1.75"
|
1
source/imgui/imstb_rectpack.h
Normal file
@ -0,0 +1 @@
|
|||||||
|
#error "Please use https://github.com/ocornut/imgui/releases/tag/v1.75"
|
1
source/imgui/imstb_textedit.h
Normal file
@ -0,0 +1 @@
|
|||||||
|
#error "Please use https://github.com/ocornut/imgui/releases/tag/v1.75"
|
1
source/imgui/imstb_truetype.h
Normal file
@ -0,0 +1 @@
|
|||||||
|
#error "Please use https://github.com/ocornut/imgui/releases/tag/v1.75"
|
107
source/ioBuffer.cpp
Normal file
@ -0,0 +1,107 @@
|
|||||||
|
// ftpd is a server implementation based on the following:
|
||||||
|
// - RFC 959 (https://tools.ietf.org/html/rfc959)
|
||||||
|
// - RFC 3659 (https://tools.ietf.org/html/rfc3659)
|
||||||
|
// - suggested implementation details from https://cr.yp.to/ftp/filesystem.html
|
||||||
|
//
|
||||||
|
// Copyright (C) 2020 Michael Theall
|
||||||
|
//
|
||||||
|
// 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 <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
#include "ioBuffer.h"
|
||||||
|
|
||||||
|
#include <cassert>
|
||||||
|
#include <cstring>
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////
|
||||||
|
IOBuffer::~IOBuffer () = default;
|
||||||
|
|
||||||
|
IOBuffer::IOBuffer (std::size_t const size_)
|
||||||
|
: m_buffer (std::make_unique<char[]> (size_)), m_size (size_)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
char *IOBuffer::freeArea () const
|
||||||
|
{
|
||||||
|
assert (m_end < m_size);
|
||||||
|
return &m_buffer[m_end];
|
||||||
|
}
|
||||||
|
|
||||||
|
std::size_t IOBuffer::freeSize () const
|
||||||
|
{
|
||||||
|
assert (m_size >= m_end);
|
||||||
|
return m_size - m_end;
|
||||||
|
}
|
||||||
|
|
||||||
|
void IOBuffer::markFree (std::size_t size_)
|
||||||
|
{
|
||||||
|
assert (m_end >= m_start);
|
||||||
|
assert (m_end - m_start >= size_);
|
||||||
|
m_start += size_;
|
||||||
|
|
||||||
|
// reset back to beginning
|
||||||
|
if (m_start == m_end)
|
||||||
|
{
|
||||||
|
m_start = 0;
|
||||||
|
m_end = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
char *IOBuffer::usedArea () const
|
||||||
|
{
|
||||||
|
assert (m_start < m_size);
|
||||||
|
return &m_buffer[m_start];
|
||||||
|
}
|
||||||
|
|
||||||
|
std::size_t IOBuffer::usedSize () const
|
||||||
|
{
|
||||||
|
assert (m_end >= m_start);
|
||||||
|
return m_end - m_start;
|
||||||
|
}
|
||||||
|
|
||||||
|
void IOBuffer::markUsed (std::size_t size_)
|
||||||
|
{
|
||||||
|
assert (m_size >= m_end);
|
||||||
|
assert (m_size - m_end >= size_);
|
||||||
|
m_end += size_;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool IOBuffer::empty () const
|
||||||
|
{
|
||||||
|
assert (m_end >= m_start);
|
||||||
|
return (m_end - m_start) == 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::size_t IOBuffer::capacity () const
|
||||||
|
{
|
||||||
|
return m_size;
|
||||||
|
}
|
||||||
|
|
||||||
|
void IOBuffer::clear ()
|
||||||
|
{
|
||||||
|
m_start = 0;
|
||||||
|
m_end = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void IOBuffer::coalesce ()
|
||||||
|
{
|
||||||
|
assert (m_size >= m_end);
|
||||||
|
assert (m_end >= m_start);
|
||||||
|
|
||||||
|
auto const size = m_end - m_start;
|
||||||
|
if (size != 0)
|
||||||
|
std::memmove (&m_buffer[0], &m_buffer[m_start], size);
|
||||||
|
|
||||||
|
m_end -= m_start;
|
||||||
|
m_start = 0;
|
||||||
|
}
|
196
source/log.cpp
Normal file
@ -0,0 +1,196 @@
|
|||||||
|
// ftpd is a server implementation based on the following:
|
||||||
|
// - RFC 959 (https://tools.ietf.org/html/rfc959)
|
||||||
|
// - RFC 3659 (https://tools.ietf.org/html/rfc3659)
|
||||||
|
// - suggested implementation details from https://cr.yp.to/ftp/filesystem.html
|
||||||
|
//
|
||||||
|
// Copyright (C) 2020 Michael Theall
|
||||||
|
//
|
||||||
|
// 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 <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
#include "log.h"
|
||||||
|
|
||||||
|
#include "imgui.h"
|
||||||
|
|
||||||
|
#include <mutex>
|
||||||
|
|
||||||
|
namespace
|
||||||
|
{
|
||||||
|
#ifdef _3DS
|
||||||
|
constexpr auto MAX_LOGS = 250;
|
||||||
|
#else
|
||||||
|
constexpr auto MAX_LOGS = 10000;
|
||||||
|
#endif
|
||||||
|
thread_local WeakLog s_log;
|
||||||
|
|
||||||
|
static char const *const s_prefix[] = {
|
||||||
|
[Log::DEBUG] = "[DEBUG]",
|
||||||
|
[Log::INFO] = "[INFO]",
|
||||||
|
[Log::ERROR] = "[ERROR]",
|
||||||
|
[Log::COMMAND] = "[COMMAND]",
|
||||||
|
[Log::RESPONSE] = "[RESPONSE]",
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////
|
||||||
|
Log::~Log () = default;
|
||||||
|
|
||||||
|
Log::Log () = default;
|
||||||
|
|
||||||
|
void Log::draw ()
|
||||||
|
{
|
||||||
|
auto const lock = std::scoped_lock (m_lock);
|
||||||
|
|
||||||
|
if (m_messages.size () > MAX_LOGS)
|
||||||
|
{
|
||||||
|
auto const begin = std::begin (m_messages);
|
||||||
|
auto const end = std::next (begin, m_messages.size () - MAX_LOGS);
|
||||||
|
m_messages.erase (begin, end);
|
||||||
|
}
|
||||||
|
|
||||||
|
static ImVec4 const s_colors[] = {
|
||||||
|
[Log::DEBUG] = ImVec4 (1.0f, 1.0f, 0.4f, 1.0f),
|
||||||
|
[Log::INFO] = ImVec4 (1.0f, 1.0f, 1.0f, 1.0f),
|
||||||
|
[Log::ERROR] = ImVec4 (1.0f, 0.4f, 0.4f, 1.0f),
|
||||||
|
[Log::COMMAND] = ImVec4 (0.4f, 1.0f, 0.4f, 1.0f),
|
||||||
|
[Log::RESPONSE] = ImVec4 (0.4f, 1.0f, 1.0f, 1.0f),
|
||||||
|
};
|
||||||
|
|
||||||
|
static char const *const s_prefix[] = {
|
||||||
|
[Log::DEBUG] = "[DEBUG]",
|
||||||
|
[Log::INFO] = "[INFO]",
|
||||||
|
[Log::ERROR] = "[ERROR]",
|
||||||
|
[Log::COMMAND] = "[COMMAND]",
|
||||||
|
[Log::RESPONSE] = "[RESPONSE]",
|
||||||
|
};
|
||||||
|
|
||||||
|
for (auto const &message : m_messages)
|
||||||
|
{
|
||||||
|
ImGui::PushStyleColor (ImGuiCol_Text, s_colors[message.level]);
|
||||||
|
ImGui::TextUnformatted (s_prefix[message.level]);
|
||||||
|
ImGui::SameLine ();
|
||||||
|
ImGui::TextUnformatted (message.message.c_str ());
|
||||||
|
ImGui::PopStyleColor ();
|
||||||
|
}
|
||||||
|
|
||||||
|
// auto scroll if scroll bar is at end
|
||||||
|
if (ImGui::GetScrollY () >= ImGui::GetScrollMaxY ())
|
||||||
|
ImGui::SetScrollHereY (1.0f);
|
||||||
|
}
|
||||||
|
|
||||||
|
SharedLog Log::create ()
|
||||||
|
{
|
||||||
|
return std::shared_ptr<Log> (new Log ());
|
||||||
|
}
|
||||||
|
|
||||||
|
void Log::bind (SharedLog log_)
|
||||||
|
{
|
||||||
|
s_log = log_;
|
||||||
|
}
|
||||||
|
|
||||||
|
void Log::debug (char const *const fmt_, ...)
|
||||||
|
{
|
||||||
|
#ifndef NDEBUG
|
||||||
|
va_list ap;
|
||||||
|
|
||||||
|
va_start (ap, fmt_);
|
||||||
|
log (DEBUG, fmt_, ap);
|
||||||
|
va_end (ap);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
void Log::info (char const *const fmt_, ...)
|
||||||
|
{
|
||||||
|
va_list ap;
|
||||||
|
|
||||||
|
va_start (ap, fmt_);
|
||||||
|
log (INFO, fmt_, ap);
|
||||||
|
va_end (ap);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Log::error (char const *const fmt_, ...)
|
||||||
|
{
|
||||||
|
va_list ap;
|
||||||
|
|
||||||
|
va_start (ap, fmt_);
|
||||||
|
log (ERROR, fmt_, ap);
|
||||||
|
va_end (ap);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Log::command (char const *const fmt_, ...)
|
||||||
|
{
|
||||||
|
va_list ap;
|
||||||
|
|
||||||
|
va_start (ap, fmt_);
|
||||||
|
log (COMMAND, fmt_, ap);
|
||||||
|
va_end (ap);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Log::response (char const *const fmt_, ...)
|
||||||
|
{
|
||||||
|
va_list ap;
|
||||||
|
|
||||||
|
va_start (ap, fmt_);
|
||||||
|
log (RESPONSE, fmt_, ap);
|
||||||
|
va_end (ap);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Log::log (Level const level_, char const *const fmt_, va_list ap_)
|
||||||
|
{
|
||||||
|
#ifdef NDEBUG
|
||||||
|
if (level_ == DEBUG)
|
||||||
|
return;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
auto log = s_log.lock ();
|
||||||
|
if (!log)
|
||||||
|
return;
|
||||||
|
|
||||||
|
thread_local static char buffer[1024];
|
||||||
|
|
||||||
|
std::vsnprintf (buffer, sizeof (buffer), fmt_, ap_);
|
||||||
|
buffer[sizeof (buffer) - 1] = '\0';
|
||||||
|
|
||||||
|
auto const lock = std::scoped_lock (log->m_lock);
|
||||||
|
#ifndef NDEBUG
|
||||||
|
std::fprintf (stderr, "%s", s_prefix[level_]);
|
||||||
|
std::fputs (buffer, stderr);
|
||||||
|
#endif
|
||||||
|
log->m_messages.emplace_back (level_, buffer);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Log::log (Level const level_, std::string_view const message_)
|
||||||
|
{
|
||||||
|
#ifdef NDEBUG
|
||||||
|
if (level_ == DEBUG)
|
||||||
|
return;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
auto log = s_log.lock ();
|
||||||
|
if (!log)
|
||||||
|
return;
|
||||||
|
|
||||||
|
auto msg = std::string (message_);
|
||||||
|
for (auto &c : msg)
|
||||||
|
{
|
||||||
|
if (c == '\0')
|
||||||
|
c = '?';
|
||||||
|
}
|
||||||
|
|
||||||
|
auto const lock = std::scoped_lock (log->m_lock);
|
||||||
|
#ifndef NDEBUG
|
||||||
|
std::fprintf (stderr, "%s", s_prefix[level_]);
|
||||||
|
std::fwrite (msg.data (), 1, msg.size (), stderr);
|
||||||
|
#endif
|
||||||
|
log->m_messages.emplace_back (level_, msg);
|
||||||
|
}
|
182
source/main.c
@ -1,182 +0,0 @@
|
|||||||
|
|
||||||
#include <errno.h>
|
|
||||||
#include <malloc.h>
|
|
||||||
#include <stdarg.h>
|
|
||||||
#include <stdio.h>
|
|
||||||
#include <unistd.h>
|
|
||||||
#ifdef _3DS
|
|
||||||
#include <3ds.h>
|
|
||||||
#elif defined(__SWITCH__)
|
|
||||||
#include <switch.h>
|
|
||||||
#endif
|
|
||||||
#include "console.h"
|
|
||||||
#include "ftp.h"
|
|
||||||
|
|
||||||
/*! looping mechanism
|
|
||||||
*
|
|
||||||
* @param[in] callback function to call during each iteration
|
|
||||||
*
|
|
||||||
* @returns loop status
|
|
||||||
*/
|
|
||||||
static loop_status_t
|
|
||||||
loop(loop_status_t (*callback)(void))
|
|
||||||
{
|
|
||||||
loop_status_t status = LOOP_CONTINUE;
|
|
||||||
|
|
||||||
#ifdef _3DS
|
|
||||||
while(aptMainLoop())
|
|
||||||
{
|
|
||||||
status = callback();
|
|
||||||
console_render();
|
|
||||||
if(status != LOOP_CONTINUE)
|
|
||||||
return status;
|
|
||||||
}
|
|
||||||
return LOOP_EXIT;
|
|
||||||
#elif defined(__SWITCH__)
|
|
||||||
while(appletMainLoop())
|
|
||||||
{
|
|
||||||
console_render();
|
|
||||||
status = callback();
|
|
||||||
if(status != LOOP_CONTINUE)
|
|
||||||
return status;
|
|
||||||
}
|
|
||||||
return LOOP_EXIT;
|
|
||||||
#else
|
|
||||||
while(status == LOOP_CONTINUE)
|
|
||||||
status = callback();
|
|
||||||
return status;
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
#ifdef _3DS
|
|
||||||
/*! wait until the B button is pressed
|
|
||||||
*
|
|
||||||
* @returns loop status
|
|
||||||
*/
|
|
||||||
static loop_status_t
|
|
||||||
wait_for_b(void)
|
|
||||||
{
|
|
||||||
/* update button state */
|
|
||||||
hidScanInput();
|
|
||||||
|
|
||||||
/* check if B was pressed */
|
|
||||||
if(hidKeysDown() & KEY_B)
|
|
||||||
return LOOP_EXIT;
|
|
||||||
|
|
||||||
/* B was not pressed */
|
|
||||||
return LOOP_CONTINUE;
|
|
||||||
}
|
|
||||||
#elif defined(__SWITCH__)
|
|
||||||
/*! wait until the B button is pressed
|
|
||||||
*
|
|
||||||
* @returns loop status
|
|
||||||
*/
|
|
||||||
static loop_status_t
|
|
||||||
wait_for_b(void)
|
|
||||||
{
|
|
||||||
/* update button state */
|
|
||||||
hidScanInput();
|
|
||||||
|
|
||||||
/* check if B was pressed */
|
|
||||||
if(hidKeysDown(CONTROLLER_P1_AUTO) & KEY_B)
|
|
||||||
return LOOP_EXIT;
|
|
||||||
|
|
||||||
/* B was not pressed */
|
|
||||||
return LOOP_CONTINUE;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/*! entry point
|
|
||||||
*
|
|
||||||
* @param[in] argc unused
|
|
||||||
* @param[in] argv unused
|
|
||||||
*
|
|
||||||
* returns exit status
|
|
||||||
*/
|
|
||||||
int
|
|
||||||
main(int argc,
|
|
||||||
char *argv[])
|
|
||||||
{
|
|
||||||
loop_status_t status = LOOP_RESTART;
|
|
||||||
|
|
||||||
#ifdef _3DS
|
|
||||||
/* initialize needed 3DS services */
|
|
||||||
acInit();
|
|
||||||
gfxInitDefault();
|
|
||||||
gfxSet3D(false);
|
|
||||||
sdmcWriteSafe(false);
|
|
||||||
#elif defined(__SWITCH__)
|
|
||||||
/* initialize needed Switch services */
|
|
||||||
nifmInitialize(NifmServiceType_User);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/* initialize console subsystem */
|
|
||||||
console_init();
|
|
||||||
|
|
||||||
#ifdef ENABLE_LOGGING
|
|
||||||
/* open log file */
|
|
||||||
#ifdef _3DS
|
|
||||||
FILE *fp = freopen("/ftpd.log", "wb", stderr);
|
|
||||||
#else
|
|
||||||
FILE *fp = freopen("ftpd.log", "wb", stderr);
|
|
||||||
#endif
|
|
||||||
if(fp == NULL)
|
|
||||||
{
|
|
||||||
console_print(RED "freopen: 0x%08X\n" RESET, errno);
|
|
||||||
goto log_fail;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* truncate log file */
|
|
||||||
if(ftruncate(fileno(fp), 0) != 0)
|
|
||||||
{
|
|
||||||
console_print(RED "ftruncate: 0x%08X\n" RESET, errno);
|
|
||||||
goto log_fail;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
console_set_status("\n" GREEN STATUS_STRING
|
|
||||||
#ifdef ENABLE_LOGGING
|
|
||||||
" DEBUG"
|
|
||||||
#endif
|
|
||||||
RESET);
|
|
||||||
|
|
||||||
while(status == LOOP_RESTART)
|
|
||||||
{
|
|
||||||
/* initialize ftp subsystem */
|
|
||||||
if(ftp_init() == 0)
|
|
||||||
{
|
|
||||||
/* ftp loop */
|
|
||||||
status = loop(ftp_loop);
|
|
||||||
|
|
||||||
/* done with ftp */
|
|
||||||
ftp_exit();
|
|
||||||
}
|
|
||||||
else
|
|
||||||
status = LOOP_EXIT;
|
|
||||||
}
|
|
||||||
|
|
||||||
#if defined(_3DS) || defined(__SWITCH__)
|
|
||||||
console_print("Press B to exit\n");
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef ENABLE_LOGGING
|
|
||||||
log_fail:
|
|
||||||
if(fclose(stderr) != 0)
|
|
||||||
console_print(RED "fclose(%d): 0x%08X\n" RESET, fileno(stderr), errno);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef _3DS
|
|
||||||
loop(wait_for_b);
|
|
||||||
|
|
||||||
/* deinitialize 3DS services */
|
|
||||||
gfxExit();
|
|
||||||
acExit();
|
|
||||||
#elif defined(__SWITCH__)
|
|
||||||
loop(wait_for_b);
|
|
||||||
|
|
||||||
/* deinitialize Switch services */
|
|
||||||
consoleExit(NULL);
|
|
||||||
nifmExit();
|
|
||||||
#endif
|
|
||||||
return 0;
|
|
||||||
}
|
|
52
source/main.cpp
Normal file
@ -0,0 +1,52 @@
|
|||||||
|
// ftpd is a server implementation based on the following:
|
||||||
|
// - RFC 959 (https://tools.ietf.org/html/rfc959)
|
||||||
|
// - RFC 3659 (https://tools.ietf.org/html/rfc3659)
|
||||||
|
// - suggested implementation details from https://cr.yp.to/ftp/filesystem.html
|
||||||
|
//
|
||||||
|
// Copyright (C) 2020 Michael Theall
|
||||||
|
//
|
||||||
|
// 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 <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
#include "platform.h"
|
||||||
|
|
||||||
|
#include "ftpServer.h"
|
||||||
|
|
||||||
|
#include "imgui.h"
|
||||||
|
|
||||||
|
#include <cstdio>
|
||||||
|
#include <cstdlib>
|
||||||
|
|
||||||
|
int main (int argc_, char *argv_[])
|
||||||
|
{
|
||||||
|
if (!platform::init ())
|
||||||
|
return EXIT_FAILURE;
|
||||||
|
|
||||||
|
auto &style = ImGui::GetStyle ();
|
||||||
|
style.WindowRounding = 0.0f;
|
||||||
|
|
||||||
|
#ifdef _3DS
|
||||||
|
style.Colors[ImGuiCol_WindowBg].w = 0.5f;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
auto server = FtpServer::create (5000);
|
||||||
|
|
||||||
|
while (platform::loop ())
|
||||||
|
{
|
||||||
|
server->draw ();
|
||||||
|
|
||||||
|
platform::render ();
|
||||||
|
}
|
||||||
|
|
||||||
|
platform::exit ();
|
||||||
|
}
|
681
source/nx/imgui_deko3d.cpp
Normal file
@ -0,0 +1,681 @@
|
|||||||
|
// ftpd is a server implementation based on the following:
|
||||||
|
// - RFC 959 (https://tools.ietf.org/html/rfc959)
|
||||||
|
// - RFC 3659 (https://tools.ietf.org/html/rfc3659)
|
||||||
|
// - suggested implementation details from https://cr.yp.to/ftp/filesystem.html
|
||||||
|
//
|
||||||
|
// Copyright (C) 2020 Michael Theall
|
||||||
|
//
|
||||||
|
// 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 <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
#include "imgui_deko3d.h"
|
||||||
|
|
||||||
|
#include "imgui.h"
|
||||||
|
|
||||||
|
#include "fs.h"
|
||||||
|
#include "log.h"
|
||||||
|
|
||||||
|
#include <deko3d.hpp>
|
||||||
|
#include <switch.h>
|
||||||
|
|
||||||
|
#include <zstd.h>
|
||||||
|
|
||||||
|
#define GLM_FORCE_DEFAULT_ALIGNED_GENTYPES
|
||||||
|
#define GLM_FORCE_INTRINSICS
|
||||||
|
#include <glm/gtc/matrix_transform.hpp>
|
||||||
|
#include <glm/gtc/type_ptr.hpp>
|
||||||
|
#include <glm/mat4x4.hpp>
|
||||||
|
|
||||||
|
#include <fcntl.h>
|
||||||
|
#include <sys/stat.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
|
||||||
|
#include <cerrno>
|
||||||
|
#include <cstdio>
|
||||||
|
#include <cstdlib>
|
||||||
|
#include <cstring>
|
||||||
|
#include <numeric>
|
||||||
|
|
||||||
|
namespace
|
||||||
|
{
|
||||||
|
constexpr auto LOGO_WIDTH = 500;
|
||||||
|
constexpr auto LOGO_HEIGHT = 493;
|
||||||
|
|
||||||
|
constexpr auto FB_NUM = 2u;
|
||||||
|
|
||||||
|
constexpr auto CMDBUF_SIZE = 1024 * 1024;
|
||||||
|
constexpr auto DATABUF_SIZE = 1024 * 1024;
|
||||||
|
constexpr auto INDEXBUF_SIZE = 1024 * 1024;
|
||||||
|
constexpr auto IMAGEBUF_SIZE = 16 * 1024 * 1024;
|
||||||
|
|
||||||
|
struct VertUBO
|
||||||
|
{
|
||||||
|
glm::mat4 projMtx;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct FragUBO
|
||||||
|
{
|
||||||
|
std::uint32_t font;
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr std::array VertexAttribState = {
|
||||||
|
// clang-format off
|
||||||
|
DkVtxAttribState{0, 0, offsetof (ImDrawVert, pos), DkVtxAttribSize_2x32, DkVtxAttribType_Float, 0},
|
||||||
|
DkVtxAttribState{0, 0, offsetof (ImDrawVert, uv), DkVtxAttribSize_2x32, DkVtxAttribType_Float, 0},
|
||||||
|
DkVtxAttribState{0, 0, offsetof (ImDrawVert, col), DkVtxAttribSize_4x8, DkVtxAttribType_Unorm, 0},
|
||||||
|
// clang-format on
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr std::array VertexBufferState = {
|
||||||
|
DkVtxBufferState{sizeof (ImDrawVert), 0},
|
||||||
|
};
|
||||||
|
|
||||||
|
dk::UniqueDevice s_device;
|
||||||
|
|
||||||
|
dk::UniqueMemBlock s_depthMemBlock;
|
||||||
|
dk::Image s_depthBuffer;
|
||||||
|
|
||||||
|
dk::UniqueMemBlock s_fbMemBlock;
|
||||||
|
dk::Image s_frameBuffers[FB_NUM];
|
||||||
|
|
||||||
|
dk::Image s_fontTexture;
|
||||||
|
dk::Image s_logoTexture;
|
||||||
|
|
||||||
|
dk::UniqueSwapchain s_swapchain;
|
||||||
|
|
||||||
|
dk::UniqueMemBlock s_codeMemBlock;
|
||||||
|
dk::Shader s_shaders[2];
|
||||||
|
|
||||||
|
dk::UniqueMemBlock s_uboMemBlock;
|
||||||
|
|
||||||
|
dk::UniqueMemBlock s_vtxMemBlock[FB_NUM];
|
||||||
|
dk::UniqueMemBlock s_idxMemBlock[FB_NUM];
|
||||||
|
dk::UniqueMemBlock s_cmdMemBlock[FB_NUM];
|
||||||
|
dk::UniqueCmdBuf s_cmdBuf[FB_NUM];
|
||||||
|
|
||||||
|
dk::UniqueMemBlock s_imageMemBlock;
|
||||||
|
dk::UniqueMemBlock s_descriptorMemBlock;
|
||||||
|
|
||||||
|
dk::UniqueQueue s_queue;
|
||||||
|
|
||||||
|
constexpr auto MAX_SAMPLERS = 1;
|
||||||
|
constexpr auto MAX_IMAGES = 2;
|
||||||
|
dk::SamplerDescriptor *s_samplerDescriptors = nullptr;
|
||||||
|
dk::ImageDescriptor *s_imageDescriptors = nullptr;
|
||||||
|
|
||||||
|
std::uintptr_t s_boundDescriptor = 0;
|
||||||
|
|
||||||
|
unsigned s_width = 0;
|
||||||
|
unsigned s_height = 0;
|
||||||
|
|
||||||
|
template <typename T, typename U>
|
||||||
|
constexpr inline std::uint32_t align (T const &size_, U const &align_)
|
||||||
|
{
|
||||||
|
return static_cast<std::uint32_t> (size_ + align_ - 1) & ~(align_ - 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
void rebuildSwapchain (unsigned const width_, unsigned const height_)
|
||||||
|
{
|
||||||
|
s_swapchain = nullptr;
|
||||||
|
|
||||||
|
dk::ImageLayout depthLayout;
|
||||||
|
dk::ImageLayoutMaker{s_device}
|
||||||
|
.setFlags (DkImageFlags_UsageRender | DkImageFlags_HwCompression)
|
||||||
|
.setFormat (DkImageFormat_Z24S8)
|
||||||
|
.setDimensions (width_, height_)
|
||||||
|
.initialize (depthLayout);
|
||||||
|
|
||||||
|
auto const depthAlign = depthLayout.getAlignment ();
|
||||||
|
auto const depthSize = depthLayout.getSize ();
|
||||||
|
|
||||||
|
if (!s_depthMemBlock)
|
||||||
|
{
|
||||||
|
s_depthMemBlock = dk::MemBlockMaker{s_device,
|
||||||
|
align (depthSize, std::max<unsigned> (depthAlign, DK_MEMBLOCK_ALIGNMENT))}
|
||||||
|
.setFlags (DkMemBlockFlags_GpuCached | DkMemBlockFlags_Image)
|
||||||
|
.create ();
|
||||||
|
}
|
||||||
|
|
||||||
|
s_depthBuffer.initialize (depthLayout, s_depthMemBlock, 0);
|
||||||
|
|
||||||
|
dk::ImageLayout fbLayout;
|
||||||
|
dk::ImageLayoutMaker{s_device}
|
||||||
|
.setFlags (
|
||||||
|
DkImageFlags_UsageRender | DkImageFlags_UsagePresent | DkImageFlags_HwCompression)
|
||||||
|
.setFormat (DkImageFormat_RGBA8_Unorm)
|
||||||
|
.setDimensions (width_, height_)
|
||||||
|
.initialize (fbLayout);
|
||||||
|
|
||||||
|
auto const fbAlign = fbLayout.getAlignment ();
|
||||||
|
auto const fbSize = fbLayout.getSize ();
|
||||||
|
|
||||||
|
if (!s_fbMemBlock)
|
||||||
|
{
|
||||||
|
s_fbMemBlock = dk::MemBlockMaker{s_device,
|
||||||
|
align (FB_NUM * fbSize, std::max<unsigned> (fbAlign, DK_MEMBLOCK_ALIGNMENT))}
|
||||||
|
.setFlags (DkMemBlockFlags_GpuCached | DkMemBlockFlags_Image)
|
||||||
|
.create ();
|
||||||
|
}
|
||||||
|
|
||||||
|
std::array<DkImage const *, FB_NUM> swapchainImages;
|
||||||
|
for (unsigned i = 0; i < FB_NUM; ++i)
|
||||||
|
{
|
||||||
|
swapchainImages[i] = &s_frameBuffers[i];
|
||||||
|
s_frameBuffers[i].initialize (fbLayout, s_fbMemBlock, i * fbSize);
|
||||||
|
}
|
||||||
|
|
||||||
|
s_swapchain = dk::SwapchainMaker{s_device, nwindowGetDefault (), swapchainImages}.create ();
|
||||||
|
}
|
||||||
|
|
||||||
|
void loadShaders ()
|
||||||
|
{
|
||||||
|
struct ShaderFile
|
||||||
|
{
|
||||||
|
ShaderFile (dk::Shader &shader_, char const *const path_)
|
||||||
|
: shader (shader_), path (path_), size (getSize (path_))
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
static std::size_t getSize (char const *const path_)
|
||||||
|
{
|
||||||
|
struct stat st;
|
||||||
|
auto const rc = stat (path_, &st);
|
||||||
|
if (rc != 0)
|
||||||
|
{
|
||||||
|
std::fprintf (stderr, "stat(%s): %s\n", path_, std::strerror (errno));
|
||||||
|
std::abort ();
|
||||||
|
}
|
||||||
|
|
||||||
|
return st.st_size;
|
||||||
|
}
|
||||||
|
|
||||||
|
dk::Shader &shader;
|
||||||
|
char const *const path;
|
||||||
|
std::size_t const size;
|
||||||
|
};
|
||||||
|
|
||||||
|
auto shaderFiles = {ShaderFile{s_shaders[0], "romfs:/shaders/imgui_vsh.dksh"},
|
||||||
|
ShaderFile{s_shaders[1], "romfs:/shaders/imgui_fsh.dksh"}};
|
||||||
|
|
||||||
|
auto const codeSize = std::accumulate (std::begin (shaderFiles),
|
||||||
|
std::end (shaderFiles),
|
||||||
|
DK_SHADER_CODE_UNUSABLE_SIZE,
|
||||||
|
[] (auto const sum_, auto const &file_) {
|
||||||
|
return sum_ + align (file_.size, DK_SHADER_CODE_ALIGNMENT);
|
||||||
|
});
|
||||||
|
|
||||||
|
s_codeMemBlock = dk::MemBlockMaker{s_device, align (codeSize, DK_MEMBLOCK_ALIGNMENT)}
|
||||||
|
.setFlags (DkMemBlockFlags_CpuUncached | DkMemBlockFlags_GpuCached |
|
||||||
|
DkMemBlockFlags_Code)
|
||||||
|
.create ();
|
||||||
|
|
||||||
|
auto const addr = static_cast<std::uint8_t *> (s_codeMemBlock.getCpuAddr ());
|
||||||
|
std::size_t offset = 0;
|
||||||
|
|
||||||
|
for (auto &file : shaderFiles)
|
||||||
|
{
|
||||||
|
std::uint32_t const codeOffset = offset;
|
||||||
|
|
||||||
|
fs::File fp;
|
||||||
|
if (!fp.open (file.path))
|
||||||
|
{
|
||||||
|
std::fprintf (stderr, "open(%s): %s\n", file.path, std::strerror (errno));
|
||||||
|
std::abort ();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!fp.readAll (&addr[offset], file.size))
|
||||||
|
{
|
||||||
|
std::fprintf (stderr, "read(%s): %s\n", file.path, std::strerror (errno));
|
||||||
|
std::abort ();
|
||||||
|
}
|
||||||
|
|
||||||
|
dk::ShaderMaker{s_codeMemBlock, codeOffset}.initialize (file.shader);
|
||||||
|
|
||||||
|
offset = align (offset + file.size, DK_SHADER_CODE_ALIGNMENT);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
DkCmdList setupRenderState (int const slot_,
|
||||||
|
ImDrawData *const drawData_,
|
||||||
|
unsigned const width_,
|
||||||
|
unsigned const height_)
|
||||||
|
{
|
||||||
|
// Setup viewport, orthographic projection matrix
|
||||||
|
// Our visible imgui space lies from drawData_->DisplayPos (top left) to
|
||||||
|
// drawData_->DisplayPos+data_data->DisplaySize (bottom right). DisplayPos is (0,0) for single
|
||||||
|
// viewport apps.
|
||||||
|
auto const L = drawData_->DisplayPos.x;
|
||||||
|
auto const R = drawData_->DisplayPos.x + drawData_->DisplaySize.x;
|
||||||
|
auto const T = drawData_->DisplayPos.y;
|
||||||
|
auto const B = drawData_->DisplayPos.y + drawData_->DisplaySize.y;
|
||||||
|
|
||||||
|
VertUBO vertUBO;
|
||||||
|
vertUBO.projMtx = glm::orthoRH_ZO (L, R, B, T, -1.0f, 1.0f);
|
||||||
|
|
||||||
|
s_cmdBuf[slot_].setViewports (0, DkViewport{0.0f, 0.0f, width_, height_});
|
||||||
|
s_cmdBuf[slot_].bindShaders (DkStageFlag_GraphicsMask, {&s_shaders[0], &s_shaders[1]});
|
||||||
|
s_cmdBuf[slot_].bindUniformBuffer (DkStage_Vertex,
|
||||||
|
0,
|
||||||
|
s_uboMemBlock.getGpuAddr (),
|
||||||
|
align (sizeof (VertUBO), DK_UNIFORM_BUF_ALIGNMENT));
|
||||||
|
s_cmdBuf[slot_].pushConstants (s_uboMemBlock.getGpuAddr (),
|
||||||
|
align (sizeof (VertUBO), DK_UNIFORM_BUF_ALIGNMENT),
|
||||||
|
0,
|
||||||
|
sizeof (VertUBO),
|
||||||
|
&vertUBO);
|
||||||
|
s_cmdBuf[slot_].bindUniformBuffer (DkStage_Fragment,
|
||||||
|
0,
|
||||||
|
s_uboMemBlock.getGpuAddr () + align (sizeof (VertUBO), DK_UNIFORM_BUF_ALIGNMENT),
|
||||||
|
align (sizeof (FragUBO), DK_UNIFORM_BUF_ALIGNMENT));
|
||||||
|
s_cmdBuf[slot_].bindRasterizerState (dk::RasterizerState{}.setCullMode (DkFace_None));
|
||||||
|
s_cmdBuf[slot_].bindColorState (dk::ColorState{}.setBlendEnable (0, true));
|
||||||
|
s_cmdBuf[slot_].bindColorWriteState (dk::ColorWriteState{});
|
||||||
|
s_cmdBuf[slot_].bindDepthStencilState (dk::DepthStencilState{}.setDepthTestEnable (false));
|
||||||
|
s_cmdBuf[slot_].bindBlendStates (0,
|
||||||
|
dk::BlendState{}.setFactors (DkBlendFactor_SrcAlpha,
|
||||||
|
DkBlendFactor_InvSrcAlpha,
|
||||||
|
DkBlendFactor_InvSrcAlpha,
|
||||||
|
DkBlendFactor_Zero));
|
||||||
|
s_cmdBuf[slot_].bindVtxAttribState (VertexAttribState);
|
||||||
|
s_cmdBuf[slot_].bindVtxBufferState (VertexBufferState);
|
||||||
|
|
||||||
|
return s_cmdBuf[slot_].finishList ();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void imgui::deko3d::init ()
|
||||||
|
{
|
||||||
|
// Setup back-end capabilities flags
|
||||||
|
ImGuiIO &io = ImGui::GetIO ();
|
||||||
|
|
||||||
|
io.BackendRendererName = "deko3d";
|
||||||
|
io.BackendFlags |= ImGuiBackendFlags_RendererHasVtxOffset;
|
||||||
|
}
|
||||||
|
|
||||||
|
void imgui::deko3d::exit ()
|
||||||
|
{
|
||||||
|
s_queue.waitIdle ();
|
||||||
|
|
||||||
|
s_queue = nullptr;
|
||||||
|
s_descriptorMemBlock = nullptr;
|
||||||
|
s_imageMemBlock = nullptr;
|
||||||
|
|
||||||
|
for (unsigned i = 0; i < FB_NUM; ++i)
|
||||||
|
{
|
||||||
|
s_cmdBuf[i] = nullptr;
|
||||||
|
s_cmdMemBlock[i] = nullptr;
|
||||||
|
s_idxMemBlock[i] = nullptr;
|
||||||
|
s_vtxMemBlock[i] = nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
s_uboMemBlock = nullptr;
|
||||||
|
s_codeMemBlock = nullptr;
|
||||||
|
s_swapchain = nullptr;
|
||||||
|
s_fbMemBlock = nullptr;
|
||||||
|
s_depthMemBlock = nullptr;
|
||||||
|
s_device = nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
void imgui::deko3d::newFrame ()
|
||||||
|
{
|
||||||
|
if (s_device)
|
||||||
|
return;
|
||||||
|
|
||||||
|
s_device = dk::DeviceMaker{}.create ();
|
||||||
|
|
||||||
|
rebuildSwapchain (1920, 1080);
|
||||||
|
|
||||||
|
loadShaders ();
|
||||||
|
|
||||||
|
s_uboMemBlock = dk::MemBlockMaker{s_device,
|
||||||
|
align (align (sizeof (VertUBO), DK_UNIFORM_BUF_ALIGNMENT) +
|
||||||
|
align (sizeof (FragUBO), DK_UNIFORM_BUF_ALIGNMENT),
|
||||||
|
DK_MEMBLOCK_ALIGNMENT)}
|
||||||
|
.setFlags (DkMemBlockFlags_CpuUncached | DkMemBlockFlags_GpuCached)
|
||||||
|
.create ();
|
||||||
|
|
||||||
|
for (std::size_t i = 0; i < FB_NUM; ++i)
|
||||||
|
{
|
||||||
|
s_vtxMemBlock[i] = dk::MemBlockMaker{s_device, align (DATABUF_SIZE, DK_MEMBLOCK_ALIGNMENT)}
|
||||||
|
.setFlags (DkMemBlockFlags_CpuUncached | DkMemBlockFlags_GpuCached)
|
||||||
|
.create ();
|
||||||
|
|
||||||
|
s_idxMemBlock[i] = dk::MemBlockMaker{s_device, align (INDEXBUF_SIZE, DK_MEMBLOCK_ALIGNMENT)}
|
||||||
|
.setFlags (DkMemBlockFlags_CpuUncached | DkMemBlockFlags_GpuCached)
|
||||||
|
.create ();
|
||||||
|
|
||||||
|
s_cmdMemBlock[i] = dk::MemBlockMaker{s_device, align (CMDBUF_SIZE, DK_MEMBLOCK_ALIGNMENT)}
|
||||||
|
.setFlags (DkMemBlockFlags_CpuUncached | DkMemBlockFlags_GpuCached)
|
||||||
|
.create ();
|
||||||
|
|
||||||
|
s_cmdBuf[i] = dk::CmdBufMaker{s_device}.create ();
|
||||||
|
|
||||||
|
s_cmdBuf[i].addMemory (s_cmdMemBlock[i], 0, s_cmdMemBlock[i].getSize ());
|
||||||
|
}
|
||||||
|
|
||||||
|
s_queue = dk::QueueMaker{s_device}.setFlags (DkQueueFlags_Graphics).create ();
|
||||||
|
|
||||||
|
s_imageMemBlock = dk::MemBlockMaker{s_device, align (IMAGEBUF_SIZE, DK_MEMBLOCK_ALIGNMENT)}
|
||||||
|
.setFlags (DkMemBlockFlags_GpuCached | DkMemBlockFlags_Image)
|
||||||
|
.create ();
|
||||||
|
|
||||||
|
// Build texture atlas
|
||||||
|
ImGuiIO &io = ImGui::GetIO ();
|
||||||
|
io.Fonts->SetTexID (nullptr);
|
||||||
|
unsigned char *pixels;
|
||||||
|
int width;
|
||||||
|
int height;
|
||||||
|
io.Fonts->GetTexDataAsAlpha8 (&pixels, &width, &height);
|
||||||
|
|
||||||
|
dk::UniqueMemBlock memBlock =
|
||||||
|
dk::MemBlockMaker{s_device, align (width * height, DK_MEMBLOCK_ALIGNMENT)}
|
||||||
|
.setFlags (DkMemBlockFlags_CpuUncached | DkMemBlockFlags_GpuCached)
|
||||||
|
.create ();
|
||||||
|
std::memcpy (memBlock.getCpuAddr (), pixels, width * height);
|
||||||
|
|
||||||
|
static_assert (sizeof (dk::ImageDescriptor) == DK_IMAGE_DESCRIPTOR_ALIGNMENT);
|
||||||
|
static_assert (sizeof (dk::SamplerDescriptor) == DK_SAMPLER_DESCRIPTOR_ALIGNMENT);
|
||||||
|
static_assert (DK_IMAGE_DESCRIPTOR_ALIGNMENT == DK_SAMPLER_DESCRIPTOR_ALIGNMENT);
|
||||||
|
s_descriptorMemBlock = dk::MemBlockMaker{s_device,
|
||||||
|
align ((MAX_SAMPLERS + MAX_IMAGES) * sizeof (dk::ImageDescriptor), DK_MEMBLOCK_ALIGNMENT)}
|
||||||
|
.setFlags (DkMemBlockFlags_CpuUncached | DkMemBlockFlags_GpuCached)
|
||||||
|
.create ();
|
||||||
|
|
||||||
|
s_samplerDescriptors =
|
||||||
|
static_cast<dk::SamplerDescriptor *> (s_descriptorMemBlock.getCpuAddr ());
|
||||||
|
s_imageDescriptors =
|
||||||
|
reinterpret_cast<dk::ImageDescriptor *> (&s_samplerDescriptors[MAX_SAMPLERS]);
|
||||||
|
|
||||||
|
s_samplerDescriptors[0].initialize (
|
||||||
|
dk::Sampler{}
|
||||||
|
.setFilter (DkFilter_Linear, DkFilter_Linear)
|
||||||
|
.setWrapMode (DkWrapMode_ClampToEdge, DkWrapMode_ClampToEdge, DkWrapMode_ClampToEdge));
|
||||||
|
|
||||||
|
auto &cmdBuf = s_cmdBuf[0];
|
||||||
|
dk::ImageLayout layout;
|
||||||
|
dk::ImageLayoutMaker{s_device}
|
||||||
|
.setFlags (0)
|
||||||
|
.setFormat (DkImageFormat_R8_Unorm)
|
||||||
|
.setDimensions (width, height)
|
||||||
|
.initialize (layout);
|
||||||
|
|
||||||
|
s_fontTexture.initialize (layout, s_imageMemBlock, 0);
|
||||||
|
s_imageDescriptors[0].initialize (s_fontTexture);
|
||||||
|
|
||||||
|
dk::ImageView imageView{s_fontTexture};
|
||||||
|
cmdBuf.copyBufferToImage ({memBlock.getGpuAddr ()}, imageView, {0, 0, 0, width, height, 1});
|
||||||
|
|
||||||
|
cmdBuf.bindSamplerDescriptorSet (s_descriptorMemBlock.getGpuAddr (), MAX_SAMPLERS);
|
||||||
|
cmdBuf.bindImageDescriptorSet (
|
||||||
|
s_descriptorMemBlock.getGpuAddr () + MAX_SAMPLERS * sizeof (dk::SamplerDescriptor),
|
||||||
|
MAX_IMAGES);
|
||||||
|
|
||||||
|
s_queue.submitCommands (cmdBuf.finishList ());
|
||||||
|
s_queue.waitIdle ();
|
||||||
|
|
||||||
|
{
|
||||||
|
auto const path = "romfs:/deko3d.rgba.zst";
|
||||||
|
|
||||||
|
struct stat st;
|
||||||
|
if (stat (path, &st) != 0)
|
||||||
|
{
|
||||||
|
std::fprintf (stderr, "stat(%s): %s\n", path, std::strerror (errno));
|
||||||
|
std::abort ();
|
||||||
|
}
|
||||||
|
|
||||||
|
fs::File fp;
|
||||||
|
if (!fp.open (path))
|
||||||
|
{
|
||||||
|
std::fprintf (stderr, "open(%s): %s\n", path, std::strerror (errno));
|
||||||
|
std::abort ();
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<char> buffer (st.st_size);
|
||||||
|
if (!fp.readAll (buffer.data (), st.st_size))
|
||||||
|
{
|
||||||
|
std::fprintf (stderr, "read(%s): %s\n", path, std::strerror (errno));
|
||||||
|
std::abort ();
|
||||||
|
}
|
||||||
|
|
||||||
|
fp.close ();
|
||||||
|
|
||||||
|
auto const size = ZSTD_getFrameContentSize (buffer.data (), st.st_size);
|
||||||
|
if (ZSTD_isError (size))
|
||||||
|
{
|
||||||
|
std::fprintf (stderr, "ZSTD_getFrameContentSize: %s\n", ZSTD_getErrorName (size));
|
||||||
|
std::abort ();
|
||||||
|
}
|
||||||
|
|
||||||
|
memBlock = dk::MemBlockMaker{s_device, align (size, DK_MEMBLOCK_ALIGNMENT)}
|
||||||
|
.setFlags (DkMemBlockFlags_CpuUncached | DkMemBlockFlags_GpuCached)
|
||||||
|
.create ();
|
||||||
|
|
||||||
|
auto const decoded =
|
||||||
|
ZSTD_decompress (memBlock.getCpuAddr (), size, buffer.data (), st.st_size);
|
||||||
|
if (ZSTD_isError (decoded))
|
||||||
|
{
|
||||||
|
std::fprintf (stderr, "ZSTD_decompress: %s\n", ZSTD_getErrorName (decoded));
|
||||||
|
std::abort ();
|
||||||
|
}
|
||||||
|
|
||||||
|
dk::ImageLayout layout;
|
||||||
|
dk::ImageLayoutMaker{s_device}
|
||||||
|
.setFlags (0)
|
||||||
|
.setFormat (DkImageFormat_RGBA8_Unorm)
|
||||||
|
.setDimensions (LOGO_WIDTH, LOGO_HEIGHT)
|
||||||
|
.initialize (layout);
|
||||||
|
|
||||||
|
auto const offset = align (width * height, DK_IMAGE_LINEAR_STRIDE_ALIGNMENT);
|
||||||
|
s_logoTexture.initialize (layout, s_imageMemBlock, offset);
|
||||||
|
s_imageDescriptors[1].initialize (s_logoTexture);
|
||||||
|
|
||||||
|
dk::ImageView imageView{s_logoTexture};
|
||||||
|
cmdBuf.copyBufferToImage (
|
||||||
|
{memBlock.getGpuAddr ()}, imageView, {0, 0, 0, LOGO_WIDTH, LOGO_HEIGHT, 1});
|
||||||
|
|
||||||
|
s_queue.submitCommands (cmdBuf.finishList ());
|
||||||
|
s_queue.waitIdle ();
|
||||||
|
}
|
||||||
|
|
||||||
|
cmdBuf.clear ();
|
||||||
|
}
|
||||||
|
|
||||||
|
void imgui::deko3d::render ()
|
||||||
|
{
|
||||||
|
auto const drawData = ImGui::GetDrawData ();
|
||||||
|
if (drawData->CmdListsCount <= 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
unsigned width = drawData->DisplaySize.x * drawData->FramebufferScale.x;
|
||||||
|
unsigned height = drawData->DisplaySize.y * drawData->FramebufferScale.y;
|
||||||
|
if (width <= 0 || height <= 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (width != s_width || height != s_height)
|
||||||
|
{
|
||||||
|
s_width = width;
|
||||||
|
s_height = height;
|
||||||
|
|
||||||
|
s_queue.waitIdle ();
|
||||||
|
rebuildSwapchain (width, height);
|
||||||
|
}
|
||||||
|
|
||||||
|
auto const slot = s_queue.acquireImage (s_swapchain);
|
||||||
|
s_cmdBuf[slot].clear ();
|
||||||
|
|
||||||
|
dk::ImageView colorTarget{s_frameBuffers[slot]};
|
||||||
|
dk::ImageView depthTarget{s_depthBuffer};
|
||||||
|
s_cmdBuf[slot].bindRenderTargets (&colorTarget, &depthTarget);
|
||||||
|
s_cmdBuf[slot].clearColor (0, DkColorMask_RGBA, 0.125f, 0.294f, 0.478f, 1.0f);
|
||||||
|
s_cmdBuf[slot].clearDepthStencil (true, 1.0f, 0xFF, 0);
|
||||||
|
s_queue.submitCommands (s_cmdBuf[slot].finishList ());
|
||||||
|
|
||||||
|
// Setup desired render state
|
||||||
|
auto const setupCmd = setupRenderState (slot, drawData, width, height);
|
||||||
|
s_queue.submitCommands (setupCmd);
|
||||||
|
|
||||||
|
s_boundDescriptor = ~static_cast<std::uintptr_t> (0);
|
||||||
|
|
||||||
|
// Will project scissor/clipping rectangles into framebuffer space
|
||||||
|
// (0,0) unless using multi-viewports
|
||||||
|
auto const clipOff = drawData->DisplayPos;
|
||||||
|
// (1,1) unless using retina display which are often (2,2)
|
||||||
|
auto const clipScale = drawData->FramebufferScale;
|
||||||
|
|
||||||
|
if (s_vtxMemBlock[slot].getSize () < drawData->TotalVtxCount * sizeof (ImDrawVert))
|
||||||
|
{
|
||||||
|
s_vtxMemBlock[slot] = nullptr;
|
||||||
|
s_vtxMemBlock[slot] =
|
||||||
|
dk::MemBlockMaker{s_device,
|
||||||
|
align (drawData->TotalVtxCount * sizeof (ImDrawVert), DK_MEMBLOCK_ALIGNMENT)}
|
||||||
|
.setFlags (DkMemBlockFlags_CpuUncached | DkMemBlockFlags_GpuCached)
|
||||||
|
.create ();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (s_idxMemBlock[slot].getSize () < drawData->TotalIdxCount * sizeof (ImDrawIdx))
|
||||||
|
{
|
||||||
|
s_idxMemBlock[slot] = nullptr;
|
||||||
|
s_idxMemBlock[slot] =
|
||||||
|
dk::MemBlockMaker{s_device,
|
||||||
|
align (drawData->TotalIdxCount * sizeof (ImDrawIdx), DK_MEMBLOCK_ALIGNMENT)}
|
||||||
|
.setFlags (DkMemBlockFlags_CpuUncached | DkMemBlockFlags_GpuCached)
|
||||||
|
.create ();
|
||||||
|
}
|
||||||
|
|
||||||
|
auto const cpuVtx = static_cast<std::uint8_t *> (s_vtxMemBlock[slot].getCpuAddr ());
|
||||||
|
auto const cpuIdx = static_cast<std::uint8_t *> (s_idxMemBlock[slot].getCpuAddr ());
|
||||||
|
|
||||||
|
auto const gpuVtx = s_vtxMemBlock[slot].getGpuAddr ();
|
||||||
|
auto const gpuIdx = s_idxMemBlock[slot].getGpuAddr ();
|
||||||
|
|
||||||
|
auto const sizeVtx = s_vtxMemBlock[slot].getSize ();
|
||||||
|
auto const sizeIdx = s_idxMemBlock[slot].getSize ();
|
||||||
|
|
||||||
|
static_assert (sizeof (ImDrawIdx) == 2);
|
||||||
|
s_cmdBuf[slot].bindVtxBuffer (0, gpuVtx, sizeVtx);
|
||||||
|
s_cmdBuf[slot].bindIdxBuffer (DkIdxFormat_Uint16, gpuIdx);
|
||||||
|
|
||||||
|
// Render command lists
|
||||||
|
std::size_t offsetVtx = 0;
|
||||||
|
std::size_t offsetIdx = 0;
|
||||||
|
for (int i = 0; i < drawData->CmdListsCount; ++i)
|
||||||
|
{
|
||||||
|
auto const &cmdList = *drawData->CmdLists[i];
|
||||||
|
|
||||||
|
auto const vtxSize = cmdList.VtxBuffer.Size * sizeof (ImDrawVert);
|
||||||
|
auto const idxSize = cmdList.IdxBuffer.Size * sizeof (ImDrawIdx);
|
||||||
|
|
||||||
|
if (sizeVtx - offsetVtx < vtxSize)
|
||||||
|
{
|
||||||
|
std::fprintf (stderr, "Not enough vertex buffer\n");
|
||||||
|
std::fprintf (stderr, "\t%zu/%u used, need %zu\n", offsetVtx, sizeVtx, vtxSize);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (sizeIdx - offsetIdx < idxSize)
|
||||||
|
{
|
||||||
|
std::fprintf (stderr, "Not enough index buffer\n");
|
||||||
|
std::fprintf (stderr, "\t%zu/%u used, need %zu\n", offsetIdx, sizeIdx, idxSize);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::memcpy (cpuVtx + offsetVtx, cmdList.VtxBuffer.Data, vtxSize);
|
||||||
|
std::memcpy (cpuIdx + offsetIdx, cmdList.IdxBuffer.Data, idxSize);
|
||||||
|
|
||||||
|
for (auto const &cmd : cmdList.CmdBuffer)
|
||||||
|
{
|
||||||
|
if (cmd.UserCallback)
|
||||||
|
{
|
||||||
|
s_queue.submitCommands (s_cmdBuf[slot].finishList ());
|
||||||
|
|
||||||
|
// User callback, registered via ImDrawList::AddCallback()
|
||||||
|
// (ImDrawCallback_ResetRenderState is a special callback value used by the user to
|
||||||
|
// request the renderer to reset render state.)
|
||||||
|
if (cmd.UserCallback == ImDrawCallback_ResetRenderState)
|
||||||
|
s_queue.submitCommands (setupCmd);
|
||||||
|
else
|
||||||
|
cmd.UserCallback (&cmdList, &cmd);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// Project scissor/clipping rectangles into framebuffer space
|
||||||
|
ImVec4 clip;
|
||||||
|
clip.x = (cmd.ClipRect.x - clipOff.x) * clipScale.x;
|
||||||
|
clip.y = (cmd.ClipRect.y - clipOff.y) * clipScale.y;
|
||||||
|
clip.z = (cmd.ClipRect.z - clipOff.x) * clipScale.x;
|
||||||
|
clip.w = (cmd.ClipRect.w - clipOff.y) * clipScale.y;
|
||||||
|
|
||||||
|
if (clip.x < width && clip.y < height && clip.z >= 0.0f && clip.w >= 0.0f)
|
||||||
|
{
|
||||||
|
if (clip.x < 0.0f)
|
||||||
|
clip.x = 0.0f;
|
||||||
|
if (clip.y < 0.0f)
|
||||||
|
clip.y = 0.0f;
|
||||||
|
|
||||||
|
s_cmdBuf[slot].setScissors (
|
||||||
|
0, DkScissor{clip.x, clip.y, clip.z - clip.x, clip.w - clip.y});
|
||||||
|
|
||||||
|
auto const descriptor = reinterpret_cast<std::uintptr_t> (cmd.TextureId);
|
||||||
|
if (descriptor >= MAX_IMAGES)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (descriptor != s_boundDescriptor)
|
||||||
|
{
|
||||||
|
s_boundDescriptor = descriptor;
|
||||||
|
|
||||||
|
s_cmdBuf[slot].bindTextures (
|
||||||
|
DkStage_Fragment, 0, dkMakeTextureHandle (descriptor, 0));
|
||||||
|
|
||||||
|
FragUBO fragUBO;
|
||||||
|
fragUBO.font = (descriptor == 0);
|
||||||
|
|
||||||
|
s_cmdBuf[slot].pushConstants (
|
||||||
|
s_uboMemBlock.getGpuAddr () +
|
||||||
|
align (sizeof (VertUBO), DK_UNIFORM_BUF_ALIGNMENT),
|
||||||
|
align (sizeof (FragUBO), DK_UNIFORM_BUF_ALIGNMENT),
|
||||||
|
0,
|
||||||
|
sizeof (FragUBO),
|
||||||
|
&fragUBO);
|
||||||
|
}
|
||||||
|
|
||||||
|
s_cmdBuf[slot].drawIndexed (DkPrimitive_Triangles,
|
||||||
|
cmd.ElemCount,
|
||||||
|
1,
|
||||||
|
cmd.IdxOffset + offsetIdx / sizeof (ImDrawIdx),
|
||||||
|
cmd.VtxOffset + offsetVtx / sizeof (ImDrawVert),
|
||||||
|
0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
offsetVtx += vtxSize;
|
||||||
|
offsetIdx += idxSize;
|
||||||
|
}
|
||||||
|
|
||||||
|
s_cmdBuf[slot].barrier (DkBarrier_Fragments, 0);
|
||||||
|
s_cmdBuf[slot].discardDepthStencil ();
|
||||||
|
s_queue.submitCommands (s_cmdBuf[slot].finishList ());
|
||||||
|
|
||||||
|
s_queue.presentImage (s_swapchain, slot);
|
||||||
|
}
|
||||||
|
|
||||||
|
void imgui::deko3d::test ()
|
||||||
|
{
|
||||||
|
auto const x1 = (s_width - LOGO_WIDTH) / 2.0f;
|
||||||
|
auto const x2 = x1 + LOGO_WIDTH;
|
||||||
|
auto const y1 = (s_height - LOGO_HEIGHT) / 2.0f;
|
||||||
|
auto const y2 = y1 + LOGO_HEIGHT;
|
||||||
|
|
||||||
|
ImGui::GetBackgroundDrawList ()->AddImage (
|
||||||
|
reinterpret_cast<ImTextureID> (1), ImVec2 (x1, y1), ImVec2 (x2, y2));
|
||||||
|
}
|
35
source/nx/imgui_deko3d.h
Normal file
@ -0,0 +1,35 @@
|
|||||||
|
// ftpd is a server implementation based on the following:
|
||||||
|
// - RFC 959 (https://tools.ietf.org/html/rfc959)
|
||||||
|
// - RFC 3659 (https://tools.ietf.org/html/rfc3659)
|
||||||
|
// - suggested implementation details from https://cr.yp.to/ftp/filesystem.html
|
||||||
|
//
|
||||||
|
// Copyright (C) 2020 Michael Theall
|
||||||
|
//
|
||||||
|
// 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 <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
namespace imgui
|
||||||
|
{
|
||||||
|
namespace deko3d
|
||||||
|
{
|
||||||
|
void init ();
|
||||||
|
void exit ();
|
||||||
|
|
||||||
|
void newFrame ();
|
||||||
|
void render ();
|
||||||
|
|
||||||
|
void test ();
|
||||||
|
}
|
||||||
|
}
|
40
source/nx/imgui_fsh.glsl
Normal file
@ -0,0 +1,40 @@
|
|||||||
|
// ftpd is a server implementation based on the following:
|
||||||
|
// - RFC 959 (https://tools.ietf.org/html/rfc959)
|
||||||
|
// - RFC 3659 (https://tools.ietf.org/html/rfc3659)
|
||||||
|
// - suggested implementation details from https://cr.yp.to/ftp/filesystem.html
|
||||||
|
//
|
||||||
|
// Copyright (C) 2020 Michael Theall
|
||||||
|
//
|
||||||
|
// 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 <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
#version 460
|
||||||
|
|
||||||
|
layout (location = 0) in vec2 vtxUv;
|
||||||
|
layout (location = 1) in vec4 vtxColor;
|
||||||
|
|
||||||
|
layout (binding = 0) uniform sampler2D tex;
|
||||||
|
|
||||||
|
layout (std140, binding = 0) uniform FragUBO {
|
||||||
|
uint font;
|
||||||
|
} ubo;
|
||||||
|
|
||||||
|
layout (location = 0) out vec4 outColor;
|
||||||
|
|
||||||
|
void main()
|
||||||
|
{
|
||||||
|
if (ubo.font != 0)
|
||||||
|
outColor = vtxColor * vec4 (vec3 (1.0), texture (tex, vtxUv).r);
|
||||||
|
else
|
||||||
|
outColor = vtxColor * texture (tex, vtxUv);
|
||||||
|
}
|
1655
source/nx/imgui_nx.cpp
Normal file
32
source/nx/imgui_nx.h
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
// ftpd is a server implementation based on the following:
|
||||||
|
// - RFC 959 (https://tools.ietf.org/html/rfc959)
|
||||||
|
// - RFC 3659 (https://tools.ietf.org/html/rfc3659)
|
||||||
|
// - suggested implementation details from https://cr.yp.to/ftp/filesystem.html
|
||||||
|
//
|
||||||
|
// Copyright (C) 2020 Michael Theall
|
||||||
|
//
|
||||||
|
// 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 <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
namespace imgui
|
||||||
|
{
|
||||||
|
namespace nx
|
||||||
|
{
|
||||||
|
bool init ();
|
||||||
|
void exit ();
|
||||||
|
|
||||||
|
void newFrame ();
|
||||||
|
}
|
||||||
|
}
|
40
source/nx/imgui_vsh.glsl
Normal file
@ -0,0 +1,40 @@
|
|||||||
|
// ftpd is a server implementation based on the following:
|
||||||
|
// - RFC 959 (https://tools.ietf.org/html/rfc959)
|
||||||
|
// - RFC 3659 (https://tools.ietf.org/html/rfc3659)
|
||||||
|
// - suggested implementation details from https://cr.yp.to/ftp/filesystem.html
|
||||||
|
//
|
||||||
|
// Copyright (C) 2020 Michael Theall
|
||||||
|
//
|
||||||
|
// 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 <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
#version 460
|
||||||
|
|
||||||
|
layout (location = 0) in vec2 inPos;
|
||||||
|
layout (location = 1) in vec2 inUv;
|
||||||
|
layout (location = 2) in vec4 inColor;
|
||||||
|
|
||||||
|
layout (location = 0) out vec2 vtxUv;
|
||||||
|
layout (location = 1) out vec4 vtxColor;
|
||||||
|
|
||||||
|
layout (std140, binding = 0) uniform VertUBO
|
||||||
|
{
|
||||||
|
mat4 projMtx;
|
||||||
|
} ubo;
|
||||||
|
|
||||||
|
void main()
|
||||||
|
{
|
||||||
|
gl_Position = ubo.projMtx * vec4 (inPos, 0.0, 1.0);
|
||||||
|
vtxUv = inUv;
|
||||||
|
vtxColor = inColor;
|
||||||
|
}
|
77
source/nx/init.c
Normal file
@ -0,0 +1,77 @@
|
|||||||
|
// ftpd is a server implementation based on the following:
|
||||||
|
// - RFC 959 (https://tools.ietf.org/html/rfc959)
|
||||||
|
// - RFC 3659 (https://tools.ietf.org/html/rfc3659)
|
||||||
|
// - suggested implementation details from https://cr.yp.to/ftp/filesystem.html
|
||||||
|
//
|
||||||
|
// Copyright (C) 2020 Michael Theall
|
||||||
|
//
|
||||||
|
// 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 <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
#include <switch.h>
|
||||||
|
|
||||||
|
#include <unistd.h>
|
||||||
|
|
||||||
|
#ifndef NDEBUG
|
||||||
|
static int s_fd = -1;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
static SocketInitConfig const s_socketInitConfig = {
|
||||||
|
.bsdsockets_version = 1,
|
||||||
|
|
||||||
|
.tcp_tx_buf_size = 1 * 1024 * 1024,
|
||||||
|
.tcp_rx_buf_size = 1 * 1024 * 1024,
|
||||||
|
.tcp_tx_buf_max_size = 4 * 1024 * 1024,
|
||||||
|
.tcp_rx_buf_max_size = 4 * 1024 * 1024,
|
||||||
|
|
||||||
|
.udp_tx_buf_size = 0x2400,
|
||||||
|
.udp_rx_buf_size = 0xA500,
|
||||||
|
|
||||||
|
.sb_efficiency = 8,
|
||||||
|
|
||||||
|
.num_bsd_sessions = 3,
|
||||||
|
.bsd_service_type = BsdServiceType_User,
|
||||||
|
};
|
||||||
|
|
||||||
|
u32 __nx_fs_num_sessions = 3;
|
||||||
|
|
||||||
|
void userAppInit ()
|
||||||
|
{
|
||||||
|
appletLockExit ();
|
||||||
|
|
||||||
|
romfsInit ();
|
||||||
|
plInitialize ();
|
||||||
|
|
||||||
|
if (R_FAILED (socketInitialize (&s_socketInitConfig)))
|
||||||
|
return;
|
||||||
|
|
||||||
|
#ifndef NDEBUG
|
||||||
|
s_fd = nxlinkStdio ();
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
void userAppExit ()
|
||||||
|
{
|
||||||
|
#ifndef NDEBUG
|
||||||
|
if (s_fd >= 0)
|
||||||
|
{
|
||||||
|
close (s_fd);
|
||||||
|
socketExit ();
|
||||||
|
s_fd = -1;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
plExit ();
|
||||||
|
romfsExit ();
|
||||||
|
appletUnlockExit ();
|
||||||
|
}
|
167
source/nx/platform.cpp
Normal file
@ -0,0 +1,167 @@
|
|||||||
|
// ftpd is a server implementation based on the following:
|
||||||
|
// - RFC 959 (https://tools.ietf.org/html/rfc959)
|
||||||
|
// - RFC 3659 (https://tools.ietf.org/html/rfc3659)
|
||||||
|
// - suggested implementation details from https://cr.yp.to/ftp/filesystem.html
|
||||||
|
//
|
||||||
|
// Copyright (C) 2020 Michael Theall
|
||||||
|
//
|
||||||
|
// 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 <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
#include "platform.h"
|
||||||
|
|
||||||
|
#include "imgui_deko3d.h"
|
||||||
|
#include "imgui_nx.h"
|
||||||
|
|
||||||
|
#include "imgui.h"
|
||||||
|
|
||||||
|
#include <switch.h>
|
||||||
|
|
||||||
|
#include <mutex>
|
||||||
|
#include <thread>
|
||||||
|
|
||||||
|
bool platform::init ()
|
||||||
|
{
|
||||||
|
IMGUI_CHECKVERSION ();
|
||||||
|
ImGui::CreateContext ();
|
||||||
|
|
||||||
|
if (!imgui::nx::init ())
|
||||||
|
{
|
||||||
|
ImGui::DestroyContext ();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
imgui::deko3d::init ();
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool platform::loop ()
|
||||||
|
{
|
||||||
|
if (!appletMainLoop ())
|
||||||
|
return false;
|
||||||
|
|
||||||
|
hidScanInput ();
|
||||||
|
|
||||||
|
auto const keysDown = hidKeysDown (CONTROLLER_P1_AUTO);
|
||||||
|
if (keysDown & KEY_PLUS)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
imgui::nx::newFrame ();
|
||||||
|
imgui::deko3d::newFrame ();
|
||||||
|
ImGui::NewFrame ();
|
||||||
|
|
||||||
|
imgui::deko3d::test ();
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void platform::render ()
|
||||||
|
{
|
||||||
|
ImGui::Render ();
|
||||||
|
|
||||||
|
imgui::deko3d::render ();
|
||||||
|
}
|
||||||
|
|
||||||
|
void platform::exit ()
|
||||||
|
{
|
||||||
|
imgui::nx::exit ();
|
||||||
|
imgui::deko3d::exit ();
|
||||||
|
|
||||||
|
ImGui::DestroyContext ();
|
||||||
|
}
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////
|
||||||
|
class platform::Thread::privateData_t
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
privateData_t () = default;
|
||||||
|
|
||||||
|
privateData_t (std::function<void ()> func_) : thread (func_)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
std::thread thread;
|
||||||
|
};
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////
|
||||||
|
platform::Thread::~Thread () = default;
|
||||||
|
|
||||||
|
platform::Thread::Thread () : m_d (new privateData_t ())
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
platform::Thread::Thread (std::function<void ()> func_) : m_d (new privateData_t (func_))
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
platform::Thread::Thread (Thread &&that_) : m_d (new privateData_t ())
|
||||||
|
{
|
||||||
|
std::swap (m_d, that_.m_d);
|
||||||
|
}
|
||||||
|
|
||||||
|
platform::Thread &platform::Thread::operator= (Thread &&that_)
|
||||||
|
{
|
||||||
|
std::swap (m_d, that_.m_d);
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
|
void platform::Thread::join ()
|
||||||
|
{
|
||||||
|
m_d->thread.join ();
|
||||||
|
}
|
||||||
|
|
||||||
|
void platform::Thread::sleep (std::chrono::milliseconds const timeout_)
|
||||||
|
{
|
||||||
|
std::this_thread::sleep_for (timeout_);
|
||||||
|
}
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////
|
||||||
|
#define USE_STD_MUTEX 1
|
||||||
|
class platform::Mutex::privateData_t
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
#if USE_STD_MUTEX
|
||||||
|
std::mutex mutex;
|
||||||
|
#else
|
||||||
|
::Mutex mutex;
|
||||||
|
#endif
|
||||||
|
};
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////
|
||||||
|
platform::Mutex::~Mutex () = default;
|
||||||
|
|
||||||
|
platform::Mutex::Mutex () : m_d (new privateData_t ())
|
||||||
|
{
|
||||||
|
#if !USE_STD_MUTEX
|
||||||
|
mutexInit (&m_d->mutex);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
void platform::Mutex::lock ()
|
||||||
|
{
|
||||||
|
#if USE_STD_MUTEX
|
||||||
|
m_d->mutex.lock ();
|
||||||
|
#else
|
||||||
|
mutexLock (&m_d->mutex);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
void platform::Mutex::unlock ()
|
||||||
|
{
|
||||||
|
#if USE_STD_MUTEX
|
||||||
|
m_d->mutex.unlock ();
|
||||||
|
#else
|
||||||
|
mutexUnlock (&m_d->mutex);
|
||||||
|
#endif
|
||||||
|
}
|
1
source/pc/KHR/khrplatform.h
Normal file
@ -0,0 +1 @@
|
|||||||
|
#error "Please use https://glad.dav1d.de/#profile=core&language=c&specification=gl&loader=on&api=gl%3D4.3"
|
1
source/pc/glad.c
Normal file
@ -0,0 +1 @@
|
|||||||
|
#error "Please use https://glad.dav1d.de/#profile=core&language=c&specification=gl&loader=on&api=gl%3D4.3"
|
1
source/pc/glad/glad.h
Normal file
@ -0,0 +1 @@
|
|||||||
|
#error "Please use https://glad.dav1d.de/#profile=core&language=c&specification=gl&loader=on&api=gl%3D4.3"
|
1
source/pc/imgui_impl_glfw.cpp
Normal file
@ -0,0 +1 @@
|
|||||||
|
#error "Please use https://github.com/ocornut/imgui"
|
1
source/pc/imgui_impl_glfw.h
Normal file
@ -0,0 +1 @@
|
|||||||
|
#error "Please use https://github.com/ocornut/imgui"
|
1
source/pc/imgui_impl_opengl3.cpp
Normal file
@ -0,0 +1 @@
|
|||||||
|
#error "Please use https://github.com/ocornut/imgui"
|
1
source/pc/imgui_impl_opengl3.h
Normal file
@ -0,0 +1 @@
|
|||||||
|
#error "Please use https://github.com/ocornut/imgui"
|
259
source/pc/platform.cpp
Normal file
@ -0,0 +1,259 @@
|
|||||||
|
// ftpd is a server implementation based on the following:
|
||||||
|
// - RFC 959 (https://tools.ietf.org/html/rfc959)
|
||||||
|
// - RFC 3659 (https://tools.ietf.org/html/rfc3659)
|
||||||
|
// - suggested implementation details from https://cr.yp.to/ftp/filesystem.html
|
||||||
|
//
|
||||||
|
// Copyright (C) 2020 Michael Theall
|
||||||
|
//
|
||||||
|
// 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 <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
#include "platform.h"
|
||||||
|
|
||||||
|
#include "imgui.h"
|
||||||
|
|
||||||
|
#include <glad/glad.h>
|
||||||
|
|
||||||
|
#include <GLFW/glfw3.h>
|
||||||
|
|
||||||
|
#include "imgui_impl_glfw.h"
|
||||||
|
#include "imgui_impl_opengl3.h"
|
||||||
|
|
||||||
|
#include <cstdio>
|
||||||
|
#include <memory>
|
||||||
|
#include <mutex>
|
||||||
|
#include <thread>
|
||||||
|
|
||||||
|
namespace
|
||||||
|
{
|
||||||
|
std::unique_ptr<GLFWwindow, void (*) (GLFWwindow *)> s_mainWindow (nullptr, glfwDestroyWindow);
|
||||||
|
|
||||||
|
void windowResize (GLFWwindow *const window_, int const width_, int const height_)
|
||||||
|
{
|
||||||
|
(void)window_;
|
||||||
|
|
||||||
|
if (!width_ || !height_)
|
||||||
|
return;
|
||||||
|
|
||||||
|
glViewport (0, 0, width_, height_);
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifndef NDEBUG
|
||||||
|
void logCallback (GLenum const source_,
|
||||||
|
GLenum const type_,
|
||||||
|
GLuint const id_,
|
||||||
|
GLenum const severity_,
|
||||||
|
GLsizei const length_,
|
||||||
|
GLchar const *const message_,
|
||||||
|
void const *const userParam_)
|
||||||
|
{
|
||||||
|
(void)source_;
|
||||||
|
(void)type_;
|
||||||
|
(void)id_;
|
||||||
|
(void)severity_;
|
||||||
|
(void)length_;
|
||||||
|
(void)userParam_;
|
||||||
|
// std::fprintf (stderr, "%s\n", message_);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
bool platform::init ()
|
||||||
|
{
|
||||||
|
if (!glfwInit ())
|
||||||
|
{
|
||||||
|
std::fprintf (stderr, "Failed to initialize GLFW\n");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
glfwWindowHint (GLFW_CONTEXT_VERSION_MAJOR, 4);
|
||||||
|
glfwWindowHint (GLFW_CONTEXT_VERSION_MINOR, 3);
|
||||||
|
glfwWindowHint (GLFW_OPENGL_PROFILE, GLFW_OPENGL_CORE_PROFILE);
|
||||||
|
#ifndef NDEBUG
|
||||||
|
glfwWindowHint (GLFW_OPENGL_DEBUG_CONTEXT, GL_TRUE);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
// set depth buffer size
|
||||||
|
glfwWindowHint (GLFW_DEPTH_BITS, 24);
|
||||||
|
glfwWindowHint (GLFW_STENCIL_BITS, 8);
|
||||||
|
|
||||||
|
s_mainWindow.reset (glfwCreateWindow (1280, 720, "Test Game", nullptr, nullptr));
|
||||||
|
if (!s_mainWindow)
|
||||||
|
{
|
||||||
|
std::fprintf (stderr, "Failed to create window\n");
|
||||||
|
glfwTerminate ();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
glfwSwapInterval (1);
|
||||||
|
|
||||||
|
// create context
|
||||||
|
glfwMakeContextCurrent (s_mainWindow.get ());
|
||||||
|
glfwSetFramebufferSizeCallback (s_mainWindow.get (), windowResize);
|
||||||
|
|
||||||
|
if (!gladLoadGL ())
|
||||||
|
{
|
||||||
|
std::fprintf (stderr, "gladLoadGL: failed\n");
|
||||||
|
platform::exit ();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifndef NDEBUG
|
||||||
|
GLint flags;
|
||||||
|
glGetIntegerv (GL_CONTEXT_FLAGS, &flags);
|
||||||
|
if (flags & GL_CONTEXT_FLAG_DEBUG_BIT)
|
||||||
|
{
|
||||||
|
glEnable (GL_DEBUG_OUTPUT);
|
||||||
|
glEnable (GL_DEBUG_OUTPUT_SYNCHRONOUS);
|
||||||
|
glDebugMessageCallback (logCallback, nullptr);
|
||||||
|
glDebugMessageControl (GL_DONT_CARE, GL_DONT_CARE, GL_DONT_CARE, 0, nullptr, GL_TRUE);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
glEnable (GL_CULL_FACE);
|
||||||
|
glFrontFace (GL_CCW);
|
||||||
|
glCullFace (GL_BACK);
|
||||||
|
|
||||||
|
glEnable (GL_DEPTH_TEST);
|
||||||
|
glDepthFunc (GL_LEQUAL);
|
||||||
|
|
||||||
|
glClearColor (104.0f / 255.0f, 176.0f / 255.0f, 216.0f / 255.0f, 1.0f);
|
||||||
|
glEnable (GL_BLEND);
|
||||||
|
glBlendFunc (GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
||||||
|
|
||||||
|
std::printf ("Renderer: %s\n", glGetString (GL_RENDERER));
|
||||||
|
std::printf ("OpenGL Version: %s\n", glGetString (GL_VERSION));
|
||||||
|
|
||||||
|
IMGUI_CHECKVERSION ();
|
||||||
|
ImGui::CreateContext ();
|
||||||
|
|
||||||
|
ImGui_ImplGlfw_InitForOpenGL (s_mainWindow.get (), true);
|
||||||
|
ImGui_ImplOpenGL3_Init ("#version 150");
|
||||||
|
|
||||||
|
ImGuiIO &io = ImGui::GetIO ();
|
||||||
|
io.IniFilename = nullptr;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool platform::loop ()
|
||||||
|
{
|
||||||
|
bool inactive;
|
||||||
|
do
|
||||||
|
{
|
||||||
|
inactive = glfwGetWindowAttrib (s_mainWindow.get (), GLFW_ICONIFIED);
|
||||||
|
(inactive ? glfwWaitEvents : glfwPollEvents) ();
|
||||||
|
|
||||||
|
if (glfwWindowShouldClose (s_mainWindow.get ()))
|
||||||
|
return false;
|
||||||
|
} while (inactive);
|
||||||
|
|
||||||
|
ImGui_ImplOpenGL3_NewFrame ();
|
||||||
|
ImGui_ImplGlfw_NewFrame ();
|
||||||
|
ImGui::NewFrame ();
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void platform::render ()
|
||||||
|
{
|
||||||
|
ImGui::Render ();
|
||||||
|
|
||||||
|
int width;
|
||||||
|
int height;
|
||||||
|
glfwGetFramebufferSize (s_mainWindow.get (), &width, &height);
|
||||||
|
glViewport (0, 0, width, height);
|
||||||
|
glClearColor (0.45f, 0.55f, 0.60f, 1.00f);
|
||||||
|
glClear (GL_COLOR_BUFFER_BIT);
|
||||||
|
|
||||||
|
ImGui_ImplOpenGL3_RenderDrawData (ImGui::GetDrawData ());
|
||||||
|
|
||||||
|
glfwSwapBuffers (s_mainWindow.get ());
|
||||||
|
}
|
||||||
|
|
||||||
|
void platform::exit ()
|
||||||
|
{
|
||||||
|
ImGui::DestroyContext ();
|
||||||
|
|
||||||
|
s_mainWindow.reset ();
|
||||||
|
glfwTerminate ();
|
||||||
|
}
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////
|
||||||
|
class platform::Thread::privateData_t
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
privateData_t () = default;
|
||||||
|
|
||||||
|
privateData_t (std::function<void ()> func_) : thread (func_)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
std::thread thread;
|
||||||
|
};
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////
|
||||||
|
platform::Thread::~Thread () = default;
|
||||||
|
|
||||||
|
platform::Thread::Thread () : m_d (new privateData_t ())
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
platform::Thread::Thread (std::function<void ()> func_) : m_d (new privateData_t (func_))
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
platform::Thread::Thread (Thread &&that_) : m_d (new privateData_t ())
|
||||||
|
{
|
||||||
|
std::swap (m_d, that_.m_d);
|
||||||
|
}
|
||||||
|
|
||||||
|
platform::Thread &platform::Thread::operator= (Thread &&that_)
|
||||||
|
{
|
||||||
|
std::swap (m_d, that_.m_d);
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
|
void platform::Thread::join ()
|
||||||
|
{
|
||||||
|
m_d->thread.join ();
|
||||||
|
}
|
||||||
|
|
||||||
|
void platform::Thread::sleep (std::chrono::milliseconds const timeout_)
|
||||||
|
{
|
||||||
|
std::this_thread::sleep_for (timeout_);
|
||||||
|
}
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////
|
||||||
|
class platform::Mutex::privateData_t
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
std::mutex mutex;
|
||||||
|
};
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////
|
||||||
|
platform::Mutex::~Mutex () = default;
|
||||||
|
|
||||||
|
platform::Mutex::Mutex () : m_d (new privateData_t ())
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
void platform::Mutex::lock ()
|
||||||
|
{
|
||||||
|
m_d->mutex.lock ();
|
||||||
|
}
|
||||||
|
|
||||||
|
void platform::Mutex::unlock ()
|
||||||
|
{
|
||||||
|
m_d->mutex.unlock ();
|
||||||
|
}
|
150
source/sockAddr.cpp
Normal file
@ -0,0 +1,150 @@
|
|||||||
|
// ftpd is a server implementation based on the following:
|
||||||
|
// - RFC 959 (https://tools.ietf.org/html/rfc959)
|
||||||
|
// - RFC 3659 (https://tools.ietf.org/html/rfc3659)
|
||||||
|
// - suggested implementation details from https://cr.yp.to/ftp/filesystem.html
|
||||||
|
//
|
||||||
|
// Copyright (C) 2020 Michael Theall
|
||||||
|
//
|
||||||
|
// 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 <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
#include "sockAddr.h"
|
||||||
|
|
||||||
|
#include <arpa/inet.h>
|
||||||
|
|
||||||
|
#include <cassert>
|
||||||
|
#include <cstring>
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////
|
||||||
|
SockAddr::~SockAddr () = default;
|
||||||
|
|
||||||
|
SockAddr::SockAddr () = default;
|
||||||
|
|
||||||
|
SockAddr::SockAddr (SockAddr const &that_) = default;
|
||||||
|
|
||||||
|
SockAddr::SockAddr (SockAddr &&that_) = default;
|
||||||
|
|
||||||
|
SockAddr &SockAddr::operator= (SockAddr const &that_) = default;
|
||||||
|
|
||||||
|
SockAddr &SockAddr::operator= (SockAddr &&that_) = default;
|
||||||
|
|
||||||
|
SockAddr::SockAddr (struct sockaddr const &addr_)
|
||||||
|
{
|
||||||
|
switch (addr_.sa_family)
|
||||||
|
{
|
||||||
|
case AF_INET:
|
||||||
|
std::memcpy (&m_addr, &addr_, sizeof (struct sockaddr_in));
|
||||||
|
break;
|
||||||
|
|
||||||
|
#ifndef _3DS
|
||||||
|
case AF_INET6:
|
||||||
|
std::memcpy (&m_addr, &addr_, sizeof (struct sockaddr_in6));
|
||||||
|
break;
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
SockAddr::SockAddr (struct sockaddr_in const &addr_)
|
||||||
|
: SockAddr (reinterpret_cast<struct sockaddr const &> (addr_))
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifndef _3DS
|
||||||
|
SockAddr::SockAddr (struct sockaddr_in6 const &addr_)
|
||||||
|
: SockAddr (reinterpret_cast<struct sockaddr const &> (addr_))
|
||||||
|
{
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
SockAddr::SockAddr (struct sockaddr_storage const &addr_)
|
||||||
|
: SockAddr (reinterpret_cast<struct sockaddr const &> (addr_))
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
SockAddr::operator struct sockaddr_in const & () const
|
||||||
|
{
|
||||||
|
assert (m_addr.ss_family == AF_INET);
|
||||||
|
return reinterpret_cast<struct sockaddr_in const &> (m_addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifndef _3DS
|
||||||
|
SockAddr::operator struct sockaddr_in6 const & () const
|
||||||
|
{
|
||||||
|
assert (m_addr.ss_family == AF_INET6);
|
||||||
|
return reinterpret_cast<struct sockaddr_in6 const &> (m_addr);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
SockAddr::operator struct sockaddr_storage const & () const
|
||||||
|
{
|
||||||
|
return m_addr;
|
||||||
|
}
|
||||||
|
|
||||||
|
SockAddr::operator struct sockaddr * ()
|
||||||
|
{
|
||||||
|
return reinterpret_cast<struct sockaddr *> (&m_addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
SockAddr::operator struct sockaddr const * () const
|
||||||
|
{
|
||||||
|
return reinterpret_cast<struct sockaddr const *> (&m_addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::uint16_t SockAddr::port () const
|
||||||
|
{
|
||||||
|
switch (m_addr.ss_family)
|
||||||
|
{
|
||||||
|
case AF_INET:
|
||||||
|
return ntohs (reinterpret_cast<struct sockaddr_in const *> (&m_addr)->sin_port);
|
||||||
|
|
||||||
|
#ifndef _3DS
|
||||||
|
case AF_INET6:
|
||||||
|
return ntohs (reinterpret_cast<struct sockaddr_in6 const *> (&m_addr)->sin6_port);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
char const *SockAddr::name (char *buffer_, std::size_t size_) const
|
||||||
|
{
|
||||||
|
switch (m_addr.ss_family)
|
||||||
|
{
|
||||||
|
case AF_INET:
|
||||||
|
return inet_ntop (AF_INET,
|
||||||
|
&reinterpret_cast<struct sockaddr_in const *> (&m_addr)->sin_addr,
|
||||||
|
buffer_,
|
||||||
|
size_);
|
||||||
|
|
||||||
|
#ifndef _3DS
|
||||||
|
case AF_INET6:
|
||||||
|
return inet_ntop (AF_INET6,
|
||||||
|
&reinterpret_cast<struct sockaddr_in6 const *> (&m_addr)->sin6_addr,
|
||||||
|
buffer_,
|
||||||
|
size_);
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
char const *SockAddr::name () const
|
||||||
|
{
|
||||||
|
#if defined(_3DS)
|
||||||
|
thread_local static char buffer[INET_ADDRSTRLEN];
|
||||||
|
#else
|
||||||
|
thread_local static char buffer[INET6_ADDRSTRLEN];
|
||||||
|
#endif
|
||||||
|
|
||||||
|
return name (buffer, sizeof (buffer));
|
||||||
|
}
|
334
source/socket.cpp
Normal file
@ -0,0 +1,334 @@
|
|||||||
|
// ftpd is a server implementation based on the following:
|
||||||
|
// - RFC 959 (https://tools.ietf.org/html/rfc959)
|
||||||
|
// - RFC 3659 (https://tools.ietf.org/html/rfc3659)
|
||||||
|
// - suggested implementation details from https://cr.yp.to/ftp/filesystem.html
|
||||||
|
//
|
||||||
|
// Copyright (C) 2020 Michael Theall
|
||||||
|
//
|
||||||
|
// 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 <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
#include "socket.h"
|
||||||
|
|
||||||
|
#include "log.h"
|
||||||
|
|
||||||
|
#include <fcntl.h>
|
||||||
|
#include <sys/socket.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
|
||||||
|
#include <poll.h>
|
||||||
|
|
||||||
|
#include <cassert>
|
||||||
|
#include <cerrno>
|
||||||
|
#include <cstdio>
|
||||||
|
#include <cstring>
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////
|
||||||
|
Socket::~Socket ()
|
||||||
|
{
|
||||||
|
if (m_listening)
|
||||||
|
Log::info ("Stop listening on [%s]:%u\n", m_sockName.name (), m_sockName.port ());
|
||||||
|
if (m_connected)
|
||||||
|
Log::info ("Closing connection to [%s]:%u\n", m_peerName.name (), m_peerName.port ());
|
||||||
|
if (::close (m_fd) != 0)
|
||||||
|
Log::error ("close: %s\n", std::strerror (errno));
|
||||||
|
}
|
||||||
|
|
||||||
|
Socket::Socket (int const fd_) : m_fd (fd_), m_listening (false), m_connected (false)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
Socket::Socket (int const fd_, SockAddr const &sockName_, SockAddr const &peerName_)
|
||||||
|
: m_sockName (sockName_),
|
||||||
|
m_peerName (peerName_),
|
||||||
|
m_fd (fd_),
|
||||||
|
m_listening (false),
|
||||||
|
m_connected (true)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
UniqueSocket Socket::accept ()
|
||||||
|
{
|
||||||
|
SockAddr addr;
|
||||||
|
socklen_t addrLen = sizeof (struct sockaddr_storage);
|
||||||
|
|
||||||
|
auto const fd = ::accept (m_fd, addr, &addrLen);
|
||||||
|
if (fd < 0)
|
||||||
|
{
|
||||||
|
Log::error ("accept: %s\n", std::strerror (errno));
|
||||||
|
return {nullptr, {}};
|
||||||
|
}
|
||||||
|
|
||||||
|
Log::info ("Accepted connection from [%s]:%u\n", addr.name (), addr.port ());
|
||||||
|
return UniqueSocket (new Socket (fd, m_sockName, addr));
|
||||||
|
}
|
||||||
|
|
||||||
|
int Socket::atMark ()
|
||||||
|
{
|
||||||
|
auto const rc = ::sockatmark (m_fd);
|
||||||
|
if (rc < 0)
|
||||||
|
Log::error ("sockatmark: %s\n", std::strerror (errno));
|
||||||
|
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool Socket::bind (SockAddr const &addr_)
|
||||||
|
{
|
||||||
|
switch (static_cast<struct sockaddr_storage const &> (addr_).ss_family)
|
||||||
|
{
|
||||||
|
case AF_INET:
|
||||||
|
if (::bind (m_fd, addr_, sizeof (struct sockaddr_in)) != 0)
|
||||||
|
{
|
||||||
|
Log::error ("bind: %s\n", std::strerror (errno));
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
#ifndef _3DS
|
||||||
|
case AF_INET6:
|
||||||
|
if (::bind (m_fd, addr_, sizeof (struct sockaddr_in6)) != 0)
|
||||||
|
{
|
||||||
|
Log::error ("bind: %s\n", std::strerror (errno));
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
default:
|
||||||
|
errno = EINVAL;
|
||||||
|
Log::error ("bind: %s\n", std::strerror (errno));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (addr_.port () == 0)
|
||||||
|
{
|
||||||
|
socklen_t addrLen = sizeof (struct sockaddr_storage);
|
||||||
|
if (::getsockname (m_fd, m_sockName, &addrLen) != 0)
|
||||||
|
Log::error ("getsockname: %s\n", std::strerror (errno));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
m_sockName = addr_;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool Socket::connect (SockAddr const &addr_)
|
||||||
|
{
|
||||||
|
if (::connect (m_fd, addr_, sizeof (struct sockaddr_storage)) != 0)
|
||||||
|
{
|
||||||
|
if (errno != EINPROGRESS)
|
||||||
|
Log::error ("connect: %s\n", std::strerror (errno));
|
||||||
|
else
|
||||||
|
{
|
||||||
|
m_peerName = addr_;
|
||||||
|
m_connected = true;
|
||||||
|
Log::info ("Connecting to [%s]:%u\n", addr_.name (), addr_.port ());
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
m_peerName = addr_;
|
||||||
|
m_connected = true;
|
||||||
|
Log::info ("Connected to [%s]:%u\n", addr_.name (), addr_.port ());
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool Socket::listen (int const backlog_)
|
||||||
|
{
|
||||||
|
if (::listen (m_fd, backlog_) != 0)
|
||||||
|
{
|
||||||
|
Log::error ("listen: %s\n", std::strerror (errno));
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
m_listening = true;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool Socket::shutdown (int const how_)
|
||||||
|
{
|
||||||
|
if (::shutdown (m_fd, how_) != 0)
|
||||||
|
{
|
||||||
|
Log::info ("shutdown: %s\n", std::strerror (errno));
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool Socket::setLinger (bool const enable_, std::chrono::seconds const time_)
|
||||||
|
{
|
||||||
|
struct linger linger;
|
||||||
|
linger.l_onoff = enable_;
|
||||||
|
linger.l_linger = time_.count ();
|
||||||
|
|
||||||
|
if (::setsockopt (m_fd, SOL_SOCKET, SO_LINGER, &linger, sizeof (linger)) != 0)
|
||||||
|
{
|
||||||
|
Log::error ("setsockopt(SO_LINGER, %s, %lus): %s\n",
|
||||||
|
enable_ ? "on" : "off",
|
||||||
|
static_cast<unsigned long> (time_.count ()),
|
||||||
|
std::strerror (errno));
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool Socket::setNonBlocking (bool const nonBlocking_)
|
||||||
|
{
|
||||||
|
auto flags = ::fcntl (m_fd, F_GETFL, 0);
|
||||||
|
if (flags == -1)
|
||||||
|
{
|
||||||
|
Log::error ("fcntl(F_GETFL): %s\n", std::strerror (errno));
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (nonBlocking_)
|
||||||
|
flags |= O_NONBLOCK;
|
||||||
|
else
|
||||||
|
flags &= ~O_NONBLOCK;
|
||||||
|
|
||||||
|
if (::fcntl (m_fd, F_SETFL, flags) != 0)
|
||||||
|
{
|
||||||
|
Log::error ("fcntl(F_SETFL): %s\n", std::strerror (errno));
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool Socket::setReuseAddress (bool const reuse_)
|
||||||
|
{
|
||||||
|
int reuse = reuse_;
|
||||||
|
if (::setsockopt (m_fd, SOL_SOCKET, SO_REUSEADDR, &reuse, sizeof (reuse)) != 0)
|
||||||
|
{
|
||||||
|
Log::error ("setsockopt(SO_REUSEADDR, %u): %s\n", reuse_, std::strerror (errno));
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool Socket::setRecvBufferSize (std::size_t const size_)
|
||||||
|
{
|
||||||
|
int size = size_;
|
||||||
|
if (::setsockopt (m_fd, SOL_SOCKET, SO_RCVBUF, &size, sizeof (size)) != 0)
|
||||||
|
{
|
||||||
|
Log::error ("setsockopt(SO_RCVBUF, %zu): %s\n", size_, std::strerror (errno));
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool Socket::setSendBufferSize (std::size_t const size_)
|
||||||
|
{
|
||||||
|
int size = size_;
|
||||||
|
if (::setsockopt (m_fd, SOL_SOCKET, SO_SNDBUF, &size, sizeof (size)) != 0)
|
||||||
|
{
|
||||||
|
Log::error ("setsockopt(SO_SNDBUF, %zu): %s\n", size_, std::strerror (errno));
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
ssize_t Socket::read (void *const buffer_, std::size_t const size_, bool const oob_)
|
||||||
|
{
|
||||||
|
assert (buffer_);
|
||||||
|
assert (size_);
|
||||||
|
auto const rc = ::recv (m_fd, buffer_, size_, oob_ ? MSG_OOB : 0);
|
||||||
|
if (rc < 0 && errno != EWOULDBLOCK)
|
||||||
|
Log::error ("recv: %s\n", std::strerror (errno));
|
||||||
|
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
|
ssize_t Socket::read (IOBuffer &buffer_, bool const oob_)
|
||||||
|
{
|
||||||
|
auto const rc = read (buffer_.freeArea (), buffer_.freeSize (), oob_);
|
||||||
|
if (rc > 0)
|
||||||
|
buffer_.markUsed (rc);
|
||||||
|
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
|
ssize_t Socket::write (void const *const buffer_, std::size_t const size_)
|
||||||
|
{
|
||||||
|
assert (buffer_);
|
||||||
|
assert (size_);
|
||||||
|
auto const rc = ::send (m_fd, buffer_, size_, 0);
|
||||||
|
if (rc < 0 && errno != EWOULDBLOCK)
|
||||||
|
Log::error ("send: %s\n", std::strerror (errno));
|
||||||
|
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
|
ssize_t Socket::write (IOBuffer &buffer_)
|
||||||
|
{
|
||||||
|
auto const rc = write (buffer_.usedArea (), buffer_.usedSize ());
|
||||||
|
if (rc > 0)
|
||||||
|
buffer_.markFree (rc);
|
||||||
|
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
|
SockAddr const &Socket::sockName () const
|
||||||
|
{
|
||||||
|
return m_sockName;
|
||||||
|
}
|
||||||
|
|
||||||
|
SockAddr const &Socket::peerName () const
|
||||||
|
{
|
||||||
|
return m_peerName;
|
||||||
|
}
|
||||||
|
|
||||||
|
UniqueSocket Socket::create ()
|
||||||
|
{
|
||||||
|
auto const fd = ::socket (AF_INET, SOCK_STREAM, 0);
|
||||||
|
if (fd < 0)
|
||||||
|
{
|
||||||
|
Log::error ("socket: %s\n", std::strerror (errno));
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
return UniqueSocket (new Socket (fd));
|
||||||
|
}
|
||||||
|
|
||||||
|
int Socket::poll (PollInfo *const info_,
|
||||||
|
std::size_t const count_,
|
||||||
|
std::chrono::milliseconds const timeout_)
|
||||||
|
{
|
||||||
|
if (count_ == 0)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
auto const pfd = std::make_unique<struct pollfd[]> (count_);
|
||||||
|
for (std::size_t i = 0; i < count_; ++i)
|
||||||
|
{
|
||||||
|
pfd[i].fd = info_[i].socket.get ().m_fd;
|
||||||
|
pfd[i].events = info_[i].events;
|
||||||
|
pfd[i].revents = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto const rc = ::poll (pfd.get (), count_, timeout_.count ());
|
||||||
|
if (rc < 0)
|
||||||
|
{
|
||||||
|
Log::error ("poll: %s\n", std::strerror (errno));
|
||||||
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (std::size_t i = 0; i < count_; ++i)
|
||||||
|
info_[i].revents = pfd[i].revents;
|
||||||
|
|
||||||
|
return rc;
|
||||||
|
}
|