Initial Commit
This commit is contained in:
commit
c13016275b
41 changed files with 3596 additions and 0 deletions
5
.gitignore
vendored
Normal file
5
.gitignore
vendored
Normal file
|
@ -0,0 +1,5 @@
|
|||
/build/
|
||||
/helper/
|
||||
/cmake-build-debug/
|
||||
.idea/
|
||||
*.iml
|
42
CMakeLists.txt
Normal file
42
CMakeLists.txt
Normal file
|
@ -0,0 +1,42 @@
|
|||
cmake_minimum_required(VERSION 3.17)
|
||||
project(VulcanoLE)
|
||||
set(CMAKE_CXX_STANDARD 20)
|
||||
set(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} -DDEBUG")
|
||||
list(APPEND CMAKE_MODULE_PATH "${CMAKE_CURRENT_SOURCE_DIR}/cmake")
|
||||
|
||||
find_package(udev REQUIRED)
|
||||
find_package(HIDAPI REQUIRED)
|
||||
pkg_check_modules(LIBUSB REQUIRED libusb-1.0)
|
||||
find_package(Threads REQUIRED)
|
||||
find_library(PULSE_FOUND NAMES pulse)
|
||||
if (PULSE_FOUND)
|
||||
set(DYNAMIC_LIBRARIES ${DYNAMIC_LIBRARIES} pulse pulse-simple)
|
||||
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -D_ENABLE_PULSE")
|
||||
set(CMAKE_LD_FLAGS "${CMAKE_LD_FLAGS} -D_ENABLE_PULSE")
|
||||
endif ()
|
||||
|
||||
set(SOURCE_FILES
|
||||
src/VulcanoLE/API/HIDHelper.cpp
|
||||
src/VulcanoLE/Keyboards/Vulcan121.cpp
|
||||
src/VulcanoLE/Audio/AudioGrabber.cpp
|
||||
src/VulcanoLE/Audio/FFT.cpp
|
||||
src/VulcanoLE/Audio/VisAudioRunner.cpp
|
||||
src/VulcanoLE/Visual/VisPlugins.cpp
|
||||
src/VulcanoLE/Scripts/Loudness.cpp
|
||||
src/VulcanoLE/Scripts/Spectrum.cpp
|
||||
src/VulcanoLE/Scripts/WeirdSpec.cpp
|
||||
)
|
||||
set(UTILS_FILES
|
||||
src/VUtils/Logging.cpp
|
||||
src/VUtils/FileHandler.cpp
|
||||
src/VUtils/Pool.cpp
|
||||
src/VUtils/Environment.cpp
|
||||
src/VUtils/StringUtils.cpp
|
||||
)
|
||||
include_directories(${CMAKE_SOURCE_DIR}/headers/)
|
||||
add_executable(
|
||||
VulcanoLE main.cpp
|
||||
${SOURCE_FILES} ${UTILS_FILES}
|
||||
)
|
||||
target_link_libraries(VulcanoLE fftw3 evdev hidapi-libusb udev ${CMAKE_DL_LIBS} ${DYNAMIC_LIBRARIES} Threads::Threads m
|
||||
debug tbb)
|
674
LICENSE
Normal file
674
LICENSE
Normal file
|
@ -0,0 +1,674 @@
|
|||
GNU GENERAL PUBLIC LICENSE
|
||||
Version 3, 29 June 2007
|
||||
|
||||
Copyright (C) 2007 Free Software Foundation, Inc. <https://fsf.org/>
|
||||
Everyone is permitted to copy and distribute verbatim copies
|
||||
of this license document, but changing it is not allowed.
|
||||
|
||||
Preamble
|
||||
|
||||
The GNU General Public License is a free, copyleft license for
|
||||
software and other kinds of works.
|
||||
|
||||
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>.
|
55
README.md
Normal file
55
README.md
Normal file
|
@ -0,0 +1,55 @@
|
|||
# VULCANO LE
|
||||
|
||||
Audio Visual Control for Vulcan121 Keyboard
|
||||
|
||||
## DEPS:
|
||||
```
|
||||
libpulse -> pipewire-pulse or pulseaudio
|
||||
cmake
|
||||
udev
|
||||
hidapi
|
||||
```
|
||||
### INSTALL
|
||||
One of Booth should be installed :) a lot of systems are using pulseaudio already
|
||||
#### ARCH
|
||||
```
|
||||
pacman -S pipewire pipewire-pulse
|
||||
OR
|
||||
pacman -S pulseaudio
|
||||
```
|
||||
## BUILD
|
||||
```
|
||||
mkdir build
|
||||
cd build
|
||||
// FOR DEBUG
|
||||
cmake -DCMAKE_BUILD_TYPE=Debug ..
|
||||
// FOR RELEASE
|
||||
make -DCMAKE_BUILD_TYPE=Release ..
|
||||
```
|
||||
|
||||
## RUN
|
||||
`./anyVulcanoLE`
|
||||
|
||||
## CONFIG
|
||||
The System should create a `state.env` file after first exit
|
||||
|
||||
Location: ~/.config/VulcanoLE/state.env
|
||||
|
||||
### EXAMPLE
|
||||
```
|
||||
audio_scale=2.0
|
||||
visual_mode=1
|
||||
shutdown_color_red=235
|
||||
shutdown_color_green=0
|
||||
shutdown_color_blue=141
|
||||
shutdown_brightness=50
|
||||
```
|
||||
|
||||
# TODO
|
||||
|
||||
- Support for custom Scripts without writing C++
|
||||
- VWeb Interface to control Configs without restarting
|
||||
- Less CPU Usage
|
||||
- Keyboard Mapping
|
||||
- Macro Support
|
||||
- Execute of Script on Hotkeys
|
228
cmake/FindHIDAPI.cmake
Normal file
228
cmake/FindHIDAPI.cmake
Normal file
|
@ -0,0 +1,228 @@
|
|||
#.rst:
|
||||
# FindHIDAPI
|
||||
# ----------
|
||||
#
|
||||
# Try to find HIDAPI library, from http://www.signal11.us/oss/hidapi/
|
||||
#
|
||||
# Cache Variables: (probably not for direct use in your Scripts)
|
||||
# HIDAPI_INCLUDE_DIR
|
||||
# HIDAPI_LIBRARY
|
||||
#
|
||||
# Non-cache variables you might use in your CMakeLists.txt:
|
||||
# HIDAPI_FOUND
|
||||
# HIDAPI_INCLUDE_DIRS
|
||||
# HIDAPI_LIBRARIES
|
||||
#
|
||||
# COMPONENTS
|
||||
# ^^^^^^^^^^
|
||||
#
|
||||
# This module respects several COMPONENTS specifying the backend you prefer:
|
||||
# ``any`` (the default), ``libusb``, and ``hidraw``.
|
||||
# The availablility of the latter two depends on your platform.
|
||||
#
|
||||
#
|
||||
# IMPORTED Targets
|
||||
# ^^^^^^^^^^^^^^^^
|
||||
|
||||
# This module defines :prop_tgt:`IMPORTED` target ``HIDAPI::hidapi`` (in all cases or
|
||||
# if no components specified), ``HIDAPI::hidapi-libusb`` (if you requested the libusb component),
|
||||
# and ``HIDAPI::hidapi-hidraw`` (if you requested the hidraw component),
|
||||
#
|
||||
# Result Variables
|
||||
# ^^^^^^^^^^^^^^^^
|
||||
#
|
||||
# ``HIDAPI_FOUND``
|
||||
# True if HIDAPI or the requested components (if any) were found.
|
||||
#
|
||||
# We recommend using the imported targets instead of the following.
|
||||
#
|
||||
# ``HIDAPI_INCLUDE_DIRS``
|
||||
# ``HIDAPI_LIBRARIES``
|
||||
#
|
||||
# Original Author:
|
||||
# 2009-2010, 2019 Ryan Pavlik <ryan.pavlik@collabora.com> <abiryan@ryand.net>
|
||||
# http://academic.cleardefinition.com
|
||||
#
|
||||
# Copyright Iowa State University 2009-2010.
|
||||
# Copyright Collabora, Ltd. 2019.
|
||||
# Distributed under the Boost Software License, Version 1.0.
|
||||
# (See accompanying file LICENSE_1_0.txt or copy at
|
||||
# http://www.boost.org/LICENSE_1_0.txt)
|
||||
|
||||
set(HIDAPI_ROOT_DIR "${HIDAPI_ROOT_DIR}" CACHE PATH "Root to search for HIDAPI")
|
||||
|
||||
# Clean up components
|
||||
if("${HIDAPI_FIND_COMPONENTS}")
|
||||
if(WIN32 OR APPLE)
|
||||
# This makes no sense on Windows or Mac, which have native APIs
|
||||
list(REMOVE "${HIDAPI_FIND_COMPONENTS}" libusb)
|
||||
endif()
|
||||
|
||||
if(NOT ${CMAKE_SYSTEM} MATCHES "Linux")
|
||||
# hidraw is only on linux
|
||||
list(REMOVE "${HIDAPI_FIND_COMPONENTS}" hidraw)
|
||||
endif()
|
||||
endif()
|
||||
if(NOT "${HIDAPI_FIND_COMPONENTS}")
|
||||
# Default to any
|
||||
set("${HIDAPI_FIND_COMPONENTS}" any)
|
||||
endif()
|
||||
|
||||
# Ask pkg-config for hints
|
||||
find_package(PkgConfig QUIET)
|
||||
if(PKG_CONFIG_FOUND)
|
||||
set(_old_prefix_path "${CMAKE_PREFIX_PATH}")
|
||||
# So pkg-config uses HIDAPI_ROOT_DIR too.
|
||||
if(HIDAPI_ROOT_DIR)
|
||||
list(APPEND CMAKE_PREFIX_PATH ${HIDAPI_ROOT_DIR})
|
||||
endif()
|
||||
pkg_check_modules(PC_HIDAPI_LIBUSB QUIET hidapi-libusb)
|
||||
pkg_check_modules(PC_HIDAPI_HIDRAW QUIET hidapi-hidraw)
|
||||
# Restore
|
||||
set(CMAKE_PREFIX_PATH "${_old_prefix_path}")
|
||||
endif()
|
||||
|
||||
# Actually search
|
||||
find_library(HIDAPI_UNDECORATED_LIBRARY
|
||||
NAMES hidapi
|
||||
PATHS
|
||||
"${HIDAPI_ROOT_DIR}"
|
||||
PATH_SUFFIXES
|
||||
lib)
|
||||
|
||||
find_library(HIDAPI_LIBUSB_LIBRARY
|
||||
NAMES hidapi hidapi-libusb
|
||||
PATHS
|
||||
"${HIDAPI_ROOT_DIR}"
|
||||
PATH_SUFFIXES
|
||||
lib
|
||||
HINTS
|
||||
${PC_HIDAPI_LIBUSB_LIBRARY_DIRS})
|
||||
|
||||
if(CMAKE_SYSTEM MATCHES "Linux")
|
||||
find_library(HIDAPI_HIDRAW_LIBRARY
|
||||
NAMES hidapi-hidraw
|
||||
HINTS
|
||||
${PC_HIDAPI_HIDRAW_LIBRARY_DIRS})
|
||||
endif()
|
||||
|
||||
find_path(HIDAPI_INCLUDE_DIR
|
||||
NAMES hidapi.h
|
||||
PATHS
|
||||
"${HIDAPI_ROOT_DIR}"
|
||||
PATH_SUFFIXES
|
||||
hidapi
|
||||
include
|
||||
include/hidapi
|
||||
HINTS
|
||||
${PC_HIDAPI_HIDRAW_INCLUDE_DIRS}
|
||||
${PC_HIDAPI_LIBUSB_INCLUDE_DIRS})
|
||||
|
||||
find_package(Threads QUIET)
|
||||
|
||||
###
|
||||
# Compute the "I don't care which backend" library
|
||||
###
|
||||
set(HIDAPI_LIBRARY)
|
||||
|
||||
# First, try to use a preferred backend if supplied
|
||||
if("${HIDAPI_FIND_COMPONENTS}" MATCHES "libusb" AND HIDAPI_LIBUSB_LIBRARY AND NOT HIDAPI_LIBRARY)
|
||||
set(HIDAPI_LIBRARY ${HIDAPI_LIBUSB_LIBRARY})
|
||||
endif()
|
||||
if("${HIDAPI_FIND_COMPONENTS}" MATCHES "hidraw" AND HIDAPI_HIDRAW_LIBRARY AND NOT HIDAPI_LIBRARY)
|
||||
set(HIDAPI_LIBRARY ${HIDAPI_HIDRAW_LIBRARY})
|
||||
endif()
|
||||
|
||||
# Then, if we don't have a preferred one, settle for anything.
|
||||
if(NOT HIDAPI_LIBRARY)
|
||||
if(HIDAPI_LIBUSB_LIBRARY)
|
||||
set(HIDAPI_LIBRARY ${HIDAPI_LIBUSB_LIBRARY})
|
||||
elseif(HIDAPI_HIDRAW_LIBRARY)
|
||||
set(HIDAPI_LIBRARY ${HIDAPI_HIDRAW_LIBRARY})
|
||||
elseif(HIDAPI_UNDECORATED_LIBRARY)
|
||||
set(HIDAPI_LIBRARY ${HIDAPI_UNDECORATED_LIBRARY})
|
||||
endif()
|
||||
endif()
|
||||
|
||||
###
|
||||
# Determine if the various requested components are found.
|
||||
###
|
||||
set(_hidapi_component_required_vars)
|
||||
|
||||
foreach(_comp IN LISTS HIDAPI_FIND_COMPONENTS)
|
||||
if("${_comp}" STREQUAL "any")
|
||||
list(APPEND _hidapi_component_required_vars
|
||||
HIDAPI_INCLUDE_DIR
|
||||
HIDAPI_LIBRARY)
|
||||
if(HIDAPI_INCLUDE_DIR AND EXISTS "${HIDAPI_LIBRARY}")
|
||||
set(HIDAPI_any_FOUND TRUE)
|
||||
mark_as_advanced(HIDAPI_INCLUDE_DIR)
|
||||
else()
|
||||
set(HIDAPI_any_FOUND FALSE)
|
||||
endif()
|
||||
|
||||
elseif("${_comp}" STREQUAL "libusb")
|
||||
list(APPEND _hidapi_component_required_vars HIDAPI_INCLUDE_DIR HIDAPI_LIBUSB_LIBRARY)
|
||||
if(HIDAPI_INCLUDE_DIR AND EXISTS "${HIDAPI_LIBUSB_LIBRARY}")
|
||||
set(HIDAPI_libusb_FOUND TRUE)
|
||||
mark_as_advanced(HIDAPI_INCLUDE_DIR HIDAPI_LIBUSB_LIBRARY)
|
||||
else()
|
||||
set(HIDAPI_libusb_FOUND FALSE)
|
||||
endif()
|
||||
|
||||
elseif("${_comp}" STREQUAL "hidraw")
|
||||
list(APPEND _hidapi_component_required_vars HIDAPI_INCLUDE_DIR HIDAPI_HIDRAW_LIBRARY)
|
||||
if(HIDAPI_INCLUDE_DIR AND EXISTS "${HIDAPI_HIDRAW_LIBRARY}")
|
||||
set(HIDAPI_hidraw_FOUND TRUE)
|
||||
mark_as_advanced(HIDAPI_INCLUDE_DIR HIDAPI_HIDRAW_LIBRARY)
|
||||
else()
|
||||
set(HIDAPI_hidraw_FOUND FALSE)
|
||||
endif()
|
||||
|
||||
else()
|
||||
message(WARNING "${_comp} is not a recognized HIDAPI component")
|
||||
set(HIDAPI_${_comp}_FOUND FALSE)
|
||||
endif()
|
||||
endforeach()
|
||||
unset(_comp)
|
||||
|
||||
###
|
||||
# FPHSA call
|
||||
###
|
||||
include(FindPackageHandleStandardArgs)
|
||||
find_package_handle_standard_args(HIDAPI
|
||||
REQUIRED_VARS
|
||||
${_hidapi_component_required_vars}
|
||||
THREADS_FOUND
|
||||
HANDLE_COMPONENTS)
|
||||
|
||||
if(HIDAPI_FOUND)
|
||||
set(HIDAPI_LIBRARIES "${HIDAPI_LIBRARY}")
|
||||
set(HIDAPI_INCLUDE_DIRS "${HIDAPI_INCLUDE_DIR}")
|
||||
if(NOT TARGET HIDAPI::hidapi)
|
||||
add_library(HIDAPI::hidapi UNKNOWN IMPORTED)
|
||||
set_target_properties(HIDAPI::hidapi PROPERTIES
|
||||
IMPORTED_LINK_INTERFACE_LANGUAGES "C"
|
||||
IMPORTED_LOCATION ${HIDAPI_LIBRARY})
|
||||
set_property(TARGET HIDAPI::hidapi PROPERTY
|
||||
IMPORTED_LINK_INTERFACE_LIBRARIES Threads::Threads)
|
||||
endif()
|
||||
endif()
|
||||
|
||||
if(HIDAPI_libusb_FOUND AND NOT TARGET HIDAPI::hidapi-libusb)
|
||||
add_library(HIDAPI::hidapi-libusb UNKNOWN IMPORTED)
|
||||
set_target_properties(HIDAPI::hidapi-libusb PROPERTIES
|
||||
IMPORTED_LINK_INTERFACE_LANGUAGES "C"
|
||||
IMPORTED_LOCATION ${HIDAPI_LIBUSB_LIBRARY})
|
||||
set_property(TARGET HIDAPI::hidapi-libusb PROPERTY
|
||||
IMPORTED_LINK_INTERFACE_LIBRARIES Threads::Threads)
|
||||
endif()
|
||||
|
||||
if(HIDAPI_hidraw_FOUND AND NOT TARGET HIDAPI::hidapi-hidraw)
|
||||
add_library(HIDAPI::hidapi-hidraw UNKNOWN IMPORTED)
|
||||
set_target_properties(HIDAPI::hidapi-hidraw PROPERTIES
|
||||
IMPORTED_LINK_INTERFACE_LANGUAGES "C"
|
||||
IMPORTED_LOCATION ${HIDAPI_HIDRAW_LIBRARY})
|
||||
set_property(TARGET HIDAPI::hidapi-hidraw PROPERTY
|
||||
IMPORTED_LINK_INTERFACE_LIBRARIES Threads::Threads)
|
||||
endif()
|
76
cmake/Findudev.cmake
Normal file
76
cmake/Findudev.cmake
Normal file
|
@ -0,0 +1,76 @@
|
|||
# - try to find the udev library
|
||||
#
|
||||
# Cache Variables: (probably not for direct use in your Scripts)
|
||||
# UDEV_INCLUDE_DIR
|
||||
# UDEV_SOURCE_DIR
|
||||
# UDEV_LIBRARY
|
||||
#
|
||||
# Non-cache variables you might use in your CMakeLists.txt:
|
||||
# UDEV_FOUND
|
||||
# UDEV_INCLUDE_DIRS
|
||||
# UDEV_LIBRARIES
|
||||
#
|
||||
# Requires these CMake modules:
|
||||
# FindPackageHandleStandardArgs (known included with CMake >=2.6.2)
|
||||
#
|
||||
# Original Author:
|
||||
# 2014 Kevin M. Godby <kevin@godby.org>
|
||||
#
|
||||
# Distributed under the Boost Software License, Version 1.0.
|
||||
# (See accompanying file LICENSE_1_0.txt or copy at
|
||||
# http://www.boost.org/LICENSE_1_0.txt)
|
||||
|
||||
set(UDEV_ROOT_DIR
|
||||
"${UDEV_ROOT_DIR}"
|
||||
CACHE
|
||||
PATH
|
||||
"Directory to search for udev")
|
||||
|
||||
find_package(PkgConfig QUIET)
|
||||
if(PKG_CONFIG_FOUND)
|
||||
pkg_check_modules(PC_LIBUDEV libudev)
|
||||
endif()
|
||||
|
||||
find_library(UDEV_LIBRARY
|
||||
NAMES
|
||||
udev
|
||||
PATHS
|
||||
${PC_LIBUDEV_LIBRARY_DIRS}
|
||||
${PC_LIBUDEV_LIBDIR}
|
||||
HINTS
|
||||
"${UDEV_ROOT_DIR}"
|
||||
PATH_SUFFIXES
|
||||
lib
|
||||
)
|
||||
|
||||
get_filename_component(_libdir "${UDEV_LIBRARY}" PATH)
|
||||
|
||||
find_path(UDEV_INCLUDE_DIR
|
||||
NAMES
|
||||
libudev.h
|
||||
PATHS
|
||||
${PC_LIBUDEV_INCLUDE_DIRS}
|
||||
${PC_LIBUDEV_INCLUDEDIR}
|
||||
HINTS
|
||||
"${_libdir}"
|
||||
"${_libdir}/.."
|
||||
"${UDEV_ROOT_DIR}"
|
||||
PATH_SUFFIXES
|
||||
include
|
||||
)
|
||||
|
||||
include(FindPackageHandleStandardArgs)
|
||||
find_package_handle_standard_args(udev
|
||||
DEFAULT_MSG
|
||||
UDEV_LIBRARY
|
||||
UDEV_INCLUDE_DIR
|
||||
)
|
||||
|
||||
if(udev_FOUND)
|
||||
list(APPEND UDEV_LIBRARIES ${UDEV_LIBRARY})
|
||||
list(APPEND UDEV_INCLUDE_DIRS ${UDEV_INCLUDE_DIR})
|
||||
mark_as_advanced(UDEV_ROOT_DIR)
|
||||
endif()
|
||||
|
||||
mark_as_advanced(UDEV_INCLUDE_DIR
|
||||
UDEV_LIBRARY)
|
166
docs.md
Normal file
166
docs.md
Normal file
|
@ -0,0 +1,166 @@
|
|||
# Notes
|
||||
Go Row first
|
||||
|
||||
## LED -> KEY (ALL WITHOUT SHIFT)
|
||||
```
|
||||
0 -> ESC
|
||||
1 -> ^
|
||||
2 -> TAB
|
||||
3 -> CAPS
|
||||
4 -> LSHIFT
|
||||
5 -> CTRL
|
||||
6 -> 1
|
||||
7 -> Q
|
||||
8 -> A
|
||||
9 -> <
|
||||
10 -> L-SUPER
|
||||
11 -> F1
|
||||
12 -> 2
|
||||
13 -> W
|
||||
14 -> S
|
||||
15 -> Y
|
||||
16 -> L-ALT
|
||||
17 -> F2
|
||||
18 -> 3
|
||||
19 -> E
|
||||
20 -> D
|
||||
21 -> X
|
||||
22 -> *EMPTY*
|
||||
23 -> F3
|
||||
24 -> 4
|
||||
25 -> R
|
||||
26 -> F
|
||||
27 -> C
|
||||
28 -> F4
|
||||
29 -> 5
|
||||
30 -> T
|
||||
31 -> G
|
||||
32 -> V
|
||||
33 -> 6
|
||||
34 -> Z
|
||||
35 -> H
|
||||
36 -> B
|
||||
37 -> SPACE
|
||||
38 -> *EMPTY*
|
||||
39 -> *EMPTY*
|
||||
40 -> *EMPTY*
|
||||
41 -> *EMPTY*
|
||||
42 -> *EMPTY*
|
||||
43 -> *EMPTY*
|
||||
44 -> *EMPTY*
|
||||
45 -> *EMPTY*
|
||||
46 -> *EMPTY*
|
||||
47 -> *EMPTY*
|
||||
48 -> F5
|
||||
49 -> 7
|
||||
50 -> U
|
||||
51 -> J
|
||||
52 -> N
|
||||
53 -> F6
|
||||
54 -> 8
|
||||
55 -> I
|
||||
56 -> K
|
||||
57 -> M
|
||||
58 -> *EMPTY*
|
||||
59 -> F7
|
||||
60 -> 9
|
||||
61 -> o
|
||||
62 -> L
|
||||
63 -> ;
|
||||
64 -> *EMPTY*
|
||||
65 -> F8
|
||||
66 -> 0
|
||||
67 -> P
|
||||
68 -> Ö
|
||||
69 -> :
|
||||
70 -> ALT+GR
|
||||
71 -> *EMPTY*
|
||||
72 -> ß
|
||||
73 -> Ü
|
||||
74 -> Ä
|
||||
75 -> -
|
||||
76 -> FN
|
||||
77 -> *EMPTY*
|
||||
78 -> F9
|
||||
79 -> ´
|
||||
80 -> +
|
||||
81 -> *EMPTY*
|
||||
82 -> RSHIFT
|
||||
83 -> R-SUPER
|
||||
84 -> F10
|
||||
85 -> F11
|
||||
86 -> F12
|
||||
87 -> BACKSPACE
|
||||
88 -> RETURN
|
||||
89 -> RCTRL
|
||||
90 -> *EMPTY*
|
||||
91 -> *EMPTY*
|
||||
92 -> *EMPTY*
|
||||
93 -> *EMPTY*
|
||||
94 -> *EMPTY*
|
||||
95 -> *EMPTY*
|
||||
96 -> #
|
||||
97 -> *EMPTY*
|
||||
98 -> *EMPTY*
|
||||
99 -> PRINT
|
||||
100 -> INS
|
||||
101 -> DEL
|
||||
102 -> ARROW LEFT
|
||||
103 -> SCROLL
|
||||
104 -> HOME
|
||||
105 -> END
|
||||
106 -> ARROW UP
|
||||
107 -> ARROW DOWN
|
||||
108 -> BREAK
|
||||
109 -> PAGE UP
|
||||
110 -> PAGE DOWN
|
||||
111 -> ARROW RIGHT
|
||||
112 -> *EMPTY*
|
||||
113 -> NUM
|
||||
114 -> NUM 7
|
||||
115 -> NUM 4
|
||||
116 -> NUM 1
|
||||
117 -> NUM 0
|
||||
118 -> *EMPTY*
|
||||
119 -> NUM /
|
||||
120 -> NUM 8
|
||||
121 -> NUM 5
|
||||
122 -> NUM 2
|
||||
123 -> *EMPTY*
|
||||
124 -> NUM *
|
||||
125 -> NUM 9
|
||||
126 -> NUM 6
|
||||
127 -> NUM 3
|
||||
128 -> NUM ,
|
||||
129 -> NUM -
|
||||
130 -> NUM +
|
||||
131 -> NUM RETURN
|
||||
132 -> *EMPTY*
|
||||
133 -> *EMPTY*
|
||||
134 -> *EMPTY*
|
||||
135 -> *EMPTY*
|
||||
136 -> *EMPTY*
|
||||
137 -> *EMPTY*
|
||||
138 -> *EMPTY*
|
||||
139 -> *EMPTY*
|
||||
140 -> *EMPTY*
|
||||
141 -> *EMPTY*
|
||||
142 -> *EMPTY*
|
||||
143 -> *EMPTY*
|
||||
```
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
30
headers/VUtils/Environment.h
Normal file
30
headers/VUtils/Environment.h
Normal file
|
@ -0,0 +1,30 @@
|
|||
#pragma once
|
||||
|
||||
#include <string>
|
||||
#include <unordered_map>
|
||||
|
||||
namespace VUtils {
|
||||
class Environment {
|
||||
public:
|
||||
Environment();
|
||||
explicit Environment(const char *filename);
|
||||
void setFile(const char *filename);
|
||||
void setPrefix(std::string prefix);
|
||||
void loadFile();
|
||||
std::string& getEnv(const std::string& name, std::string def);
|
||||
bool hasEnv(const std::string& name);
|
||||
int getAsInt(const std::string& name, int def);
|
||||
double getAsDouble(const std::string& name, double def);
|
||||
bool getAsBool(const std::string& name);
|
||||
bool saveFile();
|
||||
void set(const char* name, const char *value);
|
||||
void setNumber(const char* name, double value);
|
||||
protected:
|
||||
std::unordered_map<std::string, std::string> m_env;
|
||||
std::string m_prefix = "VENV_";
|
||||
std::string m_filename;
|
||||
};
|
||||
}
|
||||
|
||||
|
||||
|
24
headers/VUtils/FileHandler.h
Normal file
24
headers/VUtils/FileHandler.h
Normal file
|
@ -0,0 +1,24 @@
|
|||
#pragma once
|
||||
|
||||
#include <string>
|
||||
|
||||
namespace VUtils {
|
||||
class FileHandler {
|
||||
public:
|
||||
static bool fileExists(const std::string& fileName);
|
||||
static bool isDirectory(const std::string& fileName);
|
||||
static std::string readFile(const std::string& fileName);
|
||||
static bool writeFile(const std::string& fileName, const std::string& content);
|
||||
static int getFileID(const std::string& fileName);
|
||||
static void closeFD(int fd);
|
||||
static std::string getExtension(const std::string& fileName);
|
||||
static long getFileSize(int fd);
|
||||
static std::string getFileName(const std::basic_string<char>& name);
|
||||
static bool createDirectoryIfNotExist(const std::basic_string<char>& fileName);
|
||||
static char * getHomeDirectory();
|
||||
static std::string getFromHomeDir(const std::basic_string<char>& path);
|
||||
};
|
||||
}
|
||||
|
||||
|
||||
|
36
headers/VUtils/Logging.h
Normal file
36
headers/VUtils/Logging.h
Normal file
|
@ -0,0 +1,36 @@
|
|||
#pragma once
|
||||
|
||||
#include <string>
|
||||
#include <VUtils/FileHandler.h>
|
||||
|
||||
#ifdef DEBUG
|
||||
#define DEBUGLOG(message, mod) { Logging::debugMod(message, mod); }
|
||||
#define DBG(...) { Logging::debug(true,__FILE__, __FUNCTION__, __VA_ARGS__); }
|
||||
#define DBGWN(...) { Logging::debug(false,__FILE__, __FUNCTION__, __VA_ARGS__); }
|
||||
#else
|
||||
#define DEBUGLOG(message, mod)
|
||||
#define DBG(...)
|
||||
#define DBGWN(...)
|
||||
#endif
|
||||
#define ERR(...) { Logging::error(true,__FILE__, __FUNCTION__, __VA_ARGS__); }
|
||||
#define ERRWN(...) { Logging::error(false,__FILE__, __FUNCTION__, __VA_ARGS__); }
|
||||
#define LOG(...) { Logging::log(true,__FILE__, __FUNCTION__, __VA_ARGS__ ); }
|
||||
#define LOGWN(...) { Logging::log(false,__FILE__, __FUNCTION__, __VA_ARGS__ ); }
|
||||
#define WARN(...) { Logging::warn(true, __FILE__, __FUNCTION__, __VA_ARGS__ ); }
|
||||
#define WARNWN(...) { Logging::warn(false, __FILE__, __FUNCTION__, __VA_ARGS__ ); }
|
||||
|
||||
class Logging {
|
||||
public:
|
||||
enum class PrintType {
|
||||
ERROR = 0,
|
||||
LOG = 1,
|
||||
WARN = 2,
|
||||
DBG = 3
|
||||
};
|
||||
static void debug(bool newLine, const char *file, const char *func, ...) noexcept;
|
||||
static void log(bool newLine,const char *file, const char *func, ...) noexcept;
|
||||
static void warn(bool newLine,const char *file, const char *func, ...) noexcept;
|
||||
static void error(bool newLine,const char *file, const char *func, ...) noexcept;
|
||||
static std::string format(bool newLine,PrintType type, const char *log, const char *file, const char *func);
|
||||
static std::string getPrefix(PrintType type, const char *module);
|
||||
};
|
26
headers/VUtils/Pool.h
Normal file
26
headers/VUtils/Pool.h
Normal file
|
@ -0,0 +1,26 @@
|
|||
#pragma once
|
||||
|
||||
#include <functional>
|
||||
#include <thread>
|
||||
|
||||
namespace VUtils {
|
||||
struct PoolWorker {
|
||||
virtual void run() = 0;
|
||||
};
|
||||
|
||||
class Pool {
|
||||
public:
|
||||
explicit Pool(const char *name);
|
||||
~Pool();
|
||||
void setThreadCount(int count);
|
||||
void create(PoolWorker& worker);
|
||||
void joinFirstThread();
|
||||
protected:
|
||||
bool m_isCreated = false;
|
||||
int m_count = 1;
|
||||
const char *m_name = "Pool";
|
||||
PoolWorker *m_worker{};
|
||||
std::thread *m_threads{};
|
||||
void execute();
|
||||
};
|
||||
}
|
90
headers/VUtils/Storage/SafeMap.h
Normal file
90
headers/VUtils/Storage/SafeMap.h
Normal file
|
@ -0,0 +1,90 @@
|
|||
#pragma once
|
||||
#include <utility>
|
||||
#include <atomic>
|
||||
#include <queue>
|
||||
#include <condition_variable>
|
||||
#include <optional>
|
||||
#include <cassert>
|
||||
|
||||
namespace VUtils {
|
||||
template<typename T, typename S>
|
||||
struct SafeMap {
|
||||
explicit SafeMap(size_t maxSize = -1UL) : m_maxSize(maxSize), m_end(false) {};
|
||||
|
||||
bool add(const T &t, S &x);
|
||||
|
||||
bool add(T &&t, S &&x);
|
||||
|
||||
void remove(T t);
|
||||
|
||||
void clear();
|
||||
|
||||
bool has(T t);
|
||||
|
||||
S &get(T t);
|
||||
|
||||
int size();
|
||||
|
||||
private:
|
||||
std::unordered_map <T, S> m_map{};
|
||||
std::mutex m_mtx{};
|
||||
std::condition_variable m_cvFull{};
|
||||
const size_t m_maxSize{};
|
||||
std::atomic<bool> m_end{};
|
||||
};
|
||||
|
||||
template<typename T, typename S>
|
||||
bool SafeMap<T, S>::add(const T &t, S &x) {
|
||||
std::unique_lock<std::mutex> lck(m_mtx);
|
||||
while (m_map.size() == m_maxSize && !m_end) {
|
||||
return false;
|
||||
}
|
||||
assert(!m_end);
|
||||
m_map.emplace(t, std::move(x));
|
||||
return true;
|
||||
}
|
||||
|
||||
template<typename T, typename S>
|
||||
bool SafeMap<T, S>::add(T &&t, S &&x) {
|
||||
std::unique_lock<std::mutex> lck(m_mtx);
|
||||
while (m_map.size() == m_maxSize && !m_end)
|
||||
return false;
|
||||
assert(!m_end);
|
||||
m_map.push(std::move(t));
|
||||
return true;
|
||||
}
|
||||
|
||||
template<typename T, typename S>
|
||||
void SafeMap<T, S>::clear() {
|
||||
std::unique_lock <std::mutex> lck(m_mtx);
|
||||
std::unordered_map <T, S> empty;
|
||||
std::swap(m_map, empty);
|
||||
m_cvFull.notify_all();
|
||||
}
|
||||
template<typename T, typename S>
|
||||
void SafeMap<T, S>::remove(T t) {
|
||||
std::unique_lock <std::mutex> lck(m_mtx);
|
||||
if (m_map.empty() || m_end) return;
|
||||
if (m_map.contains(t)) {
|
||||
m_map.erase(t);
|
||||
}
|
||||
m_cvFull.notify_one();
|
||||
}
|
||||
|
||||
template<typename T, typename S>
|
||||
int SafeMap<T, S>::size() {
|
||||
return m_map.size();
|
||||
}
|
||||
|
||||
template<typename T, typename S>
|
||||
bool SafeMap<T, S>::has(T t) {
|
||||
std::unique_lock <std::mutex> lck(m_mtx);
|
||||
return m_map.contains(t);
|
||||
}
|
||||
|
||||
template<typename T, typename S>
|
||||
S &SafeMap<T, S>::get(T t) {
|
||||
std::unique_lock <std::mutex> lck(m_mtx);
|
||||
return m_map[t];
|
||||
}
|
||||
}
|
109
headers/VUtils/Storage/SafeQueue.h
Normal file
109
headers/VUtils/Storage/SafeQueue.h
Normal file
|
@ -0,0 +1,109 @@
|
|||
#pragma once
|
||||
|
||||
#include <utility>
|
||||
#include <atomic>
|
||||
#include <queue>
|
||||
#include <condition_variable>
|
||||
#include <optional>
|
||||
#include <cassert>
|
||||
namespace VUtils {
|
||||
template<typename T>
|
||||
struct SafeQueue {
|
||||
explicit SafeQueue(size_t maxSize = -1UL) : m_maxSize(maxSize), m_end(false) {};
|
||||
|
||||
void push(const T &t);
|
||||
|
||||
void push(T &&t);
|
||||
|
||||
void close();
|
||||
|
||||
void clear();
|
||||
|
||||
std::optional<T> pop();
|
||||
|
||||
std::optional<T> waitAndPop();
|
||||
|
||||
bool isClosed();
|
||||
|
||||
int size();
|
||||
|
||||
private:
|
||||
std::queue<T> m_que;
|
||||
std::mutex m_mtx;
|
||||
std::condition_variable m_cvEmpty, m_cvFull;
|
||||
const size_t m_maxSize;
|
||||
std::atomic<bool> m_end;
|
||||
};
|
||||
|
||||
template<typename T>
|
||||
void SafeQueue<T>::push(const T &t) {
|
||||
std::unique_lock<std::mutex> lck(m_mtx);
|
||||
while (m_que.size() == m_maxSize && !m_end) {
|
||||
// we dont wait! we return false because queue is full...
|
||||
m_cvFull.wait(lck);
|
||||
}
|
||||
assert(!m_end);
|
||||
m_que.push(std::move(t));
|
||||
m_cvEmpty.notify_one();
|
||||
}
|
||||
|
||||
template<typename T>
|
||||
void SafeQueue<T>::push(T &&t) {
|
||||
std::unique_lock<std::mutex> lck(m_mtx);
|
||||
while (m_que.size() == m_maxSize && !m_end)
|
||||
m_cvFull.wait(lck);
|
||||
assert(!m_end);
|
||||
m_que.push(std::move(t));
|
||||
m_cvEmpty.notify_one();
|
||||
}
|
||||
|
||||
template<typename T>
|
||||
void SafeQueue<T>::close() {
|
||||
m_end = true;
|
||||
std::lock_guard<std::mutex> lck(m_mtx);
|
||||
m_cvEmpty.notify_all();
|
||||
m_cvFull.notify_all();
|
||||
}
|
||||
|
||||
template<typename T>
|
||||
std::optional<T> SafeQueue<T>::pop() {
|
||||
std::unique_lock<std::mutex> lck(m_mtx);
|
||||
if (m_que.empty() || m_end) return {};
|
||||
T t = std::move(m_que.front());
|
||||
m_que.pop();
|
||||
m_cvFull.notify_one();
|
||||
return t;
|
||||
}
|
||||
|
||||
template<typename T>
|
||||
std::optional<T> SafeQueue<T>::waitAndPop() {
|
||||
std::unique_lock<std::mutex> lck(m_mtx);
|
||||
while (m_que.empty() && !m_end)
|
||||
m_cvEmpty.wait(lck);
|
||||
if (m_que.empty() || m_end) return {};
|
||||
T t = std::move(m_que.front());
|
||||
m_que.pop();
|
||||
m_cvFull.notify_one();
|
||||
return t;
|
||||
}
|
||||
|
||||
template<typename T>
|
||||
void SafeQueue<T>::clear() {
|
||||
std::unique_lock<std::mutex> lck(m_mtx);
|
||||
std::queue<T> empty;
|
||||
std::swap(m_que, empty);
|
||||
m_cvEmpty.notify_all();
|
||||
m_cvFull.notify_all();
|
||||
}
|
||||
|
||||
template<typename T>
|
||||
bool SafeQueue<T>::isClosed() {
|
||||
return m_end;
|
||||
}
|
||||
|
||||
template<typename T>
|
||||
int SafeQueue<T>::size() {
|
||||
return m_que.size();
|
||||
}
|
||||
}
|
||||
|
31
headers/VUtils/StringUtils.h
Normal file
31
headers/VUtils/StringUtils.h
Normal file
|
@ -0,0 +1,31 @@
|
|||
#pragma once
|
||||
|
||||
#include <algorithm>
|
||||
#include <string>
|
||||
|
||||
namespace VUtils {
|
||||
class StringUtils {
|
||||
public:
|
||||
static void leftTrim(std::string &s);
|
||||
|
||||
static void rightTrim(std::string &s);
|
||||
|
||||
static void trim(std::string &s);
|
||||
|
||||
static std::string leftTrimCopy(std::string s);
|
||||
|
||||
static std::string rightTrimCopy(std::string s);
|
||||
|
||||
static std::string trimCopy(std::string s);
|
||||
|
||||
static std::vector<std::string> split(const std::string &s, const std::string &delimiter);
|
||||
|
||||
static std::string urlDecode(const std::string &url);
|
||||
|
||||
static std::string urlEncode(const std::string &url);
|
||||
|
||||
static std::string join(const std::vector<std::string>& vector, const std::string& delimiter);
|
||||
|
||||
static bool hasNullByte(int size, const char string[1024]);
|
||||
};
|
||||
}// namespace VUtils
|
24
headers/VulcanoLE/API/HIDHelper.h
Normal file
24
headers/VulcanoLE/API/HIDHelper.h
Normal file
|
@ -0,0 +1,24 @@
|
|||
#pragma once
|
||||
|
||||
#include <cstdint>
|
||||
#include <linux/hidraw.h>
|
||||
#include <libudev.h>
|
||||
#include <hidapi/hidapi.h>
|
||||
|
||||
class HIDHelper {
|
||||
public:
|
||||
HIDHelper();
|
||||
~HIDHelper();
|
||||
int openDevices();
|
||||
int get_feature_report(unsigned char *buf, int size) const;
|
||||
int send_feature_report(unsigned char *buf, int size) const;
|
||||
void close_ctrl_device();
|
||||
|
||||
hid_device *m_ledDevice{};
|
||||
int ctrl_device{};
|
||||
protected:
|
||||
bool ctrlDeviceWork(udev_list_entry *cur, udev_device *usb_dev, udev_device *raw_dev, udev *udev, unsigned int product_id);
|
||||
bool findLED(hid_device_info *, unsigned int);
|
||||
|
||||
uint16_t m_products[3] = { 0x3098, 0x307a, 0x0000 };
|
||||
};
|
58
headers/VulcanoLE/Audio/AudioGrabber.h
Normal file
58
headers/VulcanoLE/Audio/AudioGrabber.h
Normal file
|
@ -0,0 +1,58 @@
|
|||
#pragma once
|
||||
|
||||
#include <pulse/error.h>
|
||||
#include <pulse/pulseaudio.h>
|
||||
#include <pulse/simple.h>
|
||||
#include <unistd.h>
|
||||
#include <string>
|
||||
#include <fstream>
|
||||
#include <memory>
|
||||
#include <mutex>
|
||||
#include <VUtils/Environment.h>
|
||||
#include <VulcanoLE/Audio/FFT.h>
|
||||
#include <VulcanoLE/Audio/Types.h>
|
||||
|
||||
static const char k_record_stream_name[] = "vulcanoLE";
|
||||
static const char k_record_stream_description[] = "Keyboard Input Stream";
|
||||
|
||||
static const int32_t k_sample_rate = 44100;
|
||||
static const int32_t k_channels = 2;
|
||||
|
||||
static const std::string k_default_monitor_postfix = ".monitor";
|
||||
|
||||
class AudioGrabber {
|
||||
public:
|
||||
enum class ReqMode {
|
||||
FFT = 0,
|
||||
RMS = 1,
|
||||
PEAK = 2,
|
||||
ALL = 3
|
||||
};
|
||||
AudioGrabber();
|
||||
~AudioGrabber();
|
||||
bool read(pcm_stereo_sample *buffer, uint32_t buffer_size);
|
||||
static AudioGrabber* createAudioGrabber();
|
||||
void init();
|
||||
FFT fft;
|
||||
ReqMode requestMode = ReqMode::FFT;
|
||||
double loudness = 0.0;
|
||||
float getLoudness();
|
||||
bool doWork();
|
||||
VUtils::Environment *env = nullptr;
|
||||
private:
|
||||
std::mutex m_mtx;
|
||||
pcm_stereo_sample *m_pcm_buffer{};
|
||||
pa_simple *m_pulseaudio_simple{};
|
||||
pa_mainloop *m_pulseaudio_mainloop{};
|
||||
std::string m_pulseaudio_default_source_name;
|
||||
void populate_default_source_name();
|
||||
bool open_pulseaudio_source(uint32_t max_buffer_size);
|
||||
static void pulseaudio_context_state_callback(pa_context *c,
|
||||
void *userdata);
|
||||
static void pulseaudio_server_info_callback(pa_context *context,
|
||||
const pa_server_info *i,
|
||||
void *userdata);
|
||||
void calculateRMS(pcm_stereo_sample *pFrame);
|
||||
void calculatePEAK(pcm_stereo_sample *pFrame);
|
||||
double m_scale = 1.0;
|
||||
};
|
24
headers/VulcanoLE/Audio/FFT.h
Normal file
24
headers/VulcanoLE/Audio/FFT.h
Normal file
|
@ -0,0 +1,24 @@
|
|||
#pragma once
|
||||
|
||||
#include <mutex>
|
||||
#include <fftw3.h>
|
||||
#include "Types.h"
|
||||
|
||||
class FFT {
|
||||
public:
|
||||
FFT();
|
||||
~FFT();
|
||||
void process(pcm_stereo_sample *pFrame);
|
||||
outputSample *getData();
|
||||
bool prepareInput(pcm_stereo_sample *buffer, uint32_t sample_size);
|
||||
protected:
|
||||
double *m_fftw_input_left{};
|
||||
double *m_fftw_input_right{};
|
||||
fftw_complex *m_fftw_output_left{};
|
||||
fftw_complex *m_fftw_output_right{};
|
||||
outputSample *m_sample = new outputSample(FFT_SIZE);
|
||||
fftw_plan m_fftw_plan_left{};
|
||||
fftw_plan m_fftw_plan_right{};
|
||||
std::mutex m_mtx;
|
||||
size_t m_fftw_results;
|
||||
};
|
23
headers/VulcanoLE/Audio/Types.h
Normal file
23
headers/VulcanoLE/Audio/Types.h
Normal file
|
@ -0,0 +1,23 @@
|
|||
#pragma once
|
||||
#define FFT_SIZE 1024
|
||||
#define BUFFER_SIZE 2048
|
||||
|
||||
struct stereo_sample_frame {
|
||||
float l;
|
||||
float r;
|
||||
};
|
||||
|
||||
using pcm_stereo_sample = struct stereo_sample_frame;
|
||||
|
||||
struct outputSample {
|
||||
explicit outputSample(int fftSize) {
|
||||
leftChannel = new double[fftSize];
|
||||
rightChannel = new double[fftSize];
|
||||
}
|
||||
~outputSample() {
|
||||
delete[] leftChannel;
|
||||
delete[] rightChannel;
|
||||
}
|
||||
double *leftChannel;
|
||||
double *rightChannel;
|
||||
};
|
20
headers/VulcanoLE/Audio/VisAudioRunner.h
Normal file
20
headers/VulcanoLE/Audio/VisAudioRunner.h
Normal file
|
@ -0,0 +1,20 @@
|
|||
#pragma once
|
||||
|
||||
#include <thread>
|
||||
#include <VulcanoLE/Audio/AudioGrabber.h>
|
||||
#include <VulcanoLE/Visual/VisPlugins.h>
|
||||
#include <VUtils/Environment.h>
|
||||
|
||||
class VisAudioRunner {
|
||||
public:
|
||||
VisAudioRunner(AudioGrabber*, VIZ::VisPlugins*);
|
||||
~VisAudioRunner();
|
||||
void init();
|
||||
void run() const;
|
||||
static VisAudioRunner* create();
|
||||
AudioGrabber* grabber;
|
||||
VIZ::VisPlugins* plugins;
|
||||
bool isActive = true;
|
||||
std::thread thread;
|
||||
VUtils::Environment *env = nullptr;
|
||||
};
|
48
headers/VulcanoLE/Keyboards/Vulcan121.h
Normal file
48
headers/VulcanoLE/Keyboards/Vulcan121.h
Normal file
|
@ -0,0 +1,48 @@
|
|||
#pragma once
|
||||
|
||||
#include <VulcanoLE/API/HIDHelper.h>
|
||||
|
||||
#define NUM_KEYS 144
|
||||
|
||||
typedef struct rgba_type {
|
||||
int16_t r{};
|
||||
int16_t g{};
|
||||
int16_t b{};
|
||||
int16_t a = 255;
|
||||
} rgba;
|
||||
|
||||
typedef struct subArray {
|
||||
int count = 0;
|
||||
int *index = nullptr; // will init in code! and returned
|
||||
} keys;
|
||||
|
||||
typedef struct led_map_type {
|
||||
rgba key[NUM_KEYS];
|
||||
} led_map;
|
||||
|
||||
class Vulcan121 {
|
||||
public:
|
||||
explicit Vulcan121(HIDHelper *helper);
|
||||
~Vulcan121() = default;
|
||||
int send_led_map(led_map *src, bool deleteMap = false);
|
||||
int send_led_to(rgba rgb);
|
||||
bool send_init_sequence();
|
||||
bool query_ctrl_report(unsigned char);
|
||||
bool send_ctrl_report(unsigned char id);
|
||||
bool wait_for_ctrl_dev();
|
||||
static led_map *createEmptyLEDMap();
|
||||
struct DATA {
|
||||
int num_rows = 6;
|
||||
int num_keys = 144;
|
||||
};
|
||||
int getColumnsForRow(int row);
|
||||
int getRowsForColumns(int col);
|
||||
int getIndex(int row, int col);
|
||||
protected:
|
||||
// we need some mapping feature! rows and cols dont have the same amount of keys. so the struct needs
|
||||
HIDHelper *m_helper;
|
||||
rgba *rv_fixed[NUM_KEYS]{};
|
||||
rgba rv_color_off = { .r = 0x0000, .g = 0x0000, .b = 0x0000 };
|
||||
keys *keyMapRow[6];
|
||||
keys *keyMapCols[0]; // need to find out the count!
|
||||
};
|
18
headers/VulcanoLE/Scripts/Loudness.h
Normal file
18
headers/VulcanoLE/Scripts/Loudness.h
Normal file
|
@ -0,0 +1,18 @@
|
|||
#pragma once
|
||||
#include <VulcanoLE/Visual/VIZ.h>
|
||||
#include <VulcanoLE/Audio/AudioGrabber.h>
|
||||
#include <VulcanoLE/Keyboards/Vulcan121.h>
|
||||
|
||||
namespace VIZ {
|
||||
class Loudness : public VIZ {
|
||||
public:
|
||||
Loudness(AudioGrabber *pGrabber, Vulcan121 *pVulcan121);
|
||||
~Loudness() override = default;
|
||||
void on_setup() override;
|
||||
void on_tick() override;
|
||||
float lastVal = 0;
|
||||
const char *name() override;
|
||||
std::string m_name = "Loudness Meter";
|
||||
};
|
||||
}
|
||||
|
22
headers/VulcanoLE/Scripts/Spectrum.h
Normal file
22
headers/VulcanoLE/Scripts/Spectrum.h
Normal file
|
@ -0,0 +1,22 @@
|
|||
#pragma once
|
||||
|
||||
#include <VulcanoLE/Visual/VIZ.h>
|
||||
#include <VulcanoLE/Audio/AudioGrabber.h>
|
||||
|
||||
namespace VIZ {
|
||||
struct Spectrum : public VIZ {
|
||||
Spectrum(AudioGrabber *pGrabber, Vulcan121 *pVulcan121);
|
||||
~Spectrum() override = default;
|
||||
void on_setup() override;
|
||||
void on_tick() override;
|
||||
double lastVal = 0;
|
||||
const char *name() override;
|
||||
std::string m_name = "Spectrum One";
|
||||
rgba colors[3] = {
|
||||
{ 0, 30, 150, 0 },
|
||||
{ 150, 150, 0, 0 },
|
||||
{ 150, 0, 40, 0 }
|
||||
};
|
||||
};
|
||||
}
|
||||
|
26
headers/VulcanoLE/Scripts/WeirdSpec.h
Normal file
26
headers/VulcanoLE/Scripts/WeirdSpec.h
Normal file
|
@ -0,0 +1,26 @@
|
|||
#pragma once
|
||||
|
||||
#include <VulcanoLE/Visual/VIZ.h>
|
||||
|
||||
namespace VIZ {
|
||||
class WeirdSpec : public VIZ {
|
||||
protected:
|
||||
int decayRate = 10;
|
||||
double lastPeak = -1;
|
||||
double threshold = 15;
|
||||
public:
|
||||
WeirdSpec(AudioGrabber *pGrabber, Vulcan121 *pVulcan121);
|
||||
~WeirdSpec() override = default;
|
||||
void on_setup() override;
|
||||
void on_tick() override;
|
||||
void switchOnPeak(double);
|
||||
int tick = 0;
|
||||
bool left = true;
|
||||
rgba colors[2] = {
|
||||
{ 0, 30, 150, 0 },
|
||||
{ 0, 150, 30, 0 }
|
||||
};
|
||||
const char *name() override;
|
||||
std::string m_name = "US Police Like Spectrum";
|
||||
};
|
||||
}
|
16
headers/VulcanoLE/Visual/VIZ.h
Normal file
16
headers/VulcanoLE/Visual/VIZ.h
Normal file
|
@ -0,0 +1,16 @@
|
|||
#pragma once
|
||||
#include <VulcanoLE/Audio/AudioGrabber.h>
|
||||
#include <VulcanoLE/Keyboards/Vulcan121.h>
|
||||
|
||||
namespace VIZ {
|
||||
struct VIZ {
|
||||
VIZ(AudioGrabber *pGrabber, Vulcan121 *pVulcan121) : grabber(pGrabber), keyboard(pVulcan121) {}
|
||||
virtual ~VIZ() = default;
|
||||
virtual void on_setup() = 0;
|
||||
virtual void on_tick() = 0;
|
||||
Vulcan121 *keyboard{};
|
||||
Vulcan121::DATA keyboardData = Vulcan121::DATA{};
|
||||
AudioGrabber *grabber{};
|
||||
virtual const char *name() = 0;
|
||||
};
|
||||
}
|
25
headers/VulcanoLE/Visual/VisPlugins.h
Normal file
25
headers/VulcanoLE/Visual/VisPlugins.h
Normal file
|
@ -0,0 +1,25 @@
|
|||
#pragma once
|
||||
|
||||
#include <vector>
|
||||
#include <VulcanoLE/Scripts/Spectrum.h>
|
||||
|
||||
#define VIZSIZE 3
|
||||
|
||||
namespace VIZ {
|
||||
struct VisPlugins {
|
||||
int mode = 0;
|
||||
void init(HIDHelper *, AudioGrabber*);
|
||||
void on_startup();
|
||||
void on_tick();
|
||||
void on_shutdown();
|
||||
void setCurrentMode(int);
|
||||
~VisPlugins();
|
||||
VUtils::Environment *env;
|
||||
protected:
|
||||
VIZ *viz[VIZSIZE]{};
|
||||
VIZ *currentVis;
|
||||
Vulcan121 *keyboard;
|
||||
AudioGrabber *grabber;
|
||||
};
|
||||
}
|
||||
|
311
layout.md
Normal file
311
layout.md
Normal file
|
@ -0,0 +1,311 @@
|
|||
|
||||
# VULCANO LED MAPPING LAYOUT
|
||||
|
||||
Empty Keys: 39
|
||||
|
||||
Key Count With LEDs: 105
|
||||
|
||||
## INDEX -> KEY
|
||||
|
||||
| Index | Key |
|
||||
|--------|--------------|
|
||||
| 0 | ESC |
|
||||
| 1 | ^ |
|
||||
| 2 | TAB |
|
||||
| 3 | CAPS |
|
||||
| 4 | LSHIFT |
|
||||
| 5 | CTRL |
|
||||
| 6 | 1 |
|
||||
| 7 | Q |
|
||||
| 8 | A |
|
||||
| 9 | < |
|
||||
| 10 | L-SUPER |
|
||||
| 11 | F1 |
|
||||
| 12 | 2 |
|
||||
| 13 | W |
|
||||
| 14 | S |
|
||||
| 15 | Y |
|
||||
| 16 | L-ALT |
|
||||
| 17 | F2 |
|
||||
| 18 | 3 |
|
||||
| 19 | E |
|
||||
| 20 | D |
|
||||
| 21 | X |
|
||||
| 22 | *EMPTY* |
|
||||
| 23 | F3 |
|
||||
| 24 | 4 |
|
||||
| 25 | R |
|
||||
| 26 | F |
|
||||
| 27 | C |
|
||||
| 28 | F4 |
|
||||
| 29 | 5 |
|
||||
| 30 | T |
|
||||
| 31 | G |
|
||||
| 32 | V |
|
||||
| 33 | 6 |
|
||||
| 34 | Z |
|
||||
| 35 | H |
|
||||
| 36 | B |
|
||||
| 37 | SPACE |
|
||||
| 38 | *EMPTY* |
|
||||
| 39 | *EMPTY* |
|
||||
| 40 | *EMPTY* |
|
||||
| 41 | *EMPTY* |
|
||||
| 42 | *EMPTY* |
|
||||
| 43 | *EMPTY* |
|
||||
| 44 | *EMPTY* |
|
||||
| 45 | *EMPTY* |
|
||||
| 46 | *EMPTY* |
|
||||
| 47 | *EMPTY* |
|
||||
| 48 | F5 |
|
||||
| 49 | 7 |
|
||||
| 50 | U |
|
||||
| 51 | J |
|
||||
| 52 | N |
|
||||
| 53 | F6 |
|
||||
| 54 | 8 |
|
||||
| 55 | I |
|
||||
| 56 | K |
|
||||
| 57 | M |
|
||||
| 58 | *EMPTY* |
|
||||
| 59 | F7 |
|
||||
| 60 | 9 |
|
||||
| 61 | o |
|
||||
| 62 | L |
|
||||
| 63 | ; |
|
||||
| 64 | *EMPTY* |
|
||||
| 65 | F8 |
|
||||
| 66 | 0 |
|
||||
| 67 | P |
|
||||
| 68 | Ö |
|
||||
| 69 | : |
|
||||
| 70 | ALT+GR |
|
||||
| 71 | *EMPTY* |
|
||||
| 72 | ß |
|
||||
| 73 | Ü |
|
||||
| 74 | Ä |
|
||||
| 75 | - |
|
||||
| 76 | FN |
|
||||
| 77 | *EMPTY* |
|
||||
| 78 | F9 |
|
||||
| 79 | ´ |
|
||||
| 80 | + |
|
||||
| 81 | *EMPTY* |
|
||||
| 82 | RSHIFT |
|
||||
| 83 | R-SUPER |
|
||||
| 84 | F10 |
|
||||
| 85 | F11 |
|
||||
| 86 | F12 |
|
||||
| 87 | BACKSPACE |
|
||||
| 88 | RETURN |
|
||||
| 89 | RCTRL |
|
||||
| 90 | *EMPTY* |
|
||||
| 91 | *EMPTY* |
|
||||
| 92 | *EMPTY* |
|
||||
| 93 | *EMPTY* |
|
||||
| 94 | *EMPTY* |
|
||||
| 95 | *EMPTY* |
|
||||
| 96 | # |
|
||||
| 97 | *EMPTY* |
|
||||
| 98 | *EMPTY* |
|
||||
| 99 | PRINT |
|
||||
| 100 | INS |
|
||||
| 101 | DEL |
|
||||
| 102 | ARROW LEFT |
|
||||
| 103 | SCROLL |
|
||||
| 104 | HOME |
|
||||
| 105 | END |
|
||||
| 106 | ARROW UP |
|
||||
| 107 | ARROW DOWN |
|
||||
| 108 | BREAK |
|
||||
| 109 | PAGE UP |
|
||||
| 110 | PAGE DOWN |
|
||||
| 111 | ARROW RIGHT |
|
||||
| 112 | *EMPTY* |
|
||||
| 113 | NUM |
|
||||
| 114 | NUM 7 |
|
||||
| 115 | NUM 4 |
|
||||
| 116 | NUM 1 |
|
||||
| 117 | NUM 0 |
|
||||
| 118 | *EMPTY* |
|
||||
| 119 | NUM / |
|
||||
| 120 | NUM 8 |
|
||||
| 121 | NUM 5 |
|
||||
| 122 | NUM 2 |
|
||||
| 123 | *EMPTY* |
|
||||
| 124 | NUM * |
|
||||
| 125 | NUM 9 |
|
||||
| 126 | NUM 6 |
|
||||
| 127 | NUM 3 |
|
||||
| 128 | NUM , |
|
||||
| 129 | NUM - |
|
||||
| 130 | NUM + |
|
||||
| 131 | NUM RETURN |
|
||||
| 132 | *EMPTY* |
|
||||
| 133 | *EMPTY* |
|
||||
| 134 | *EMPTY* |
|
||||
| 135 | *EMPTY* |
|
||||
| 136 | *EMPTY* |
|
||||
| 137 | *EMPTY* |
|
||||
| 138 | *EMPTY* |
|
||||
| 139 | *EMPTY* |
|
||||
| 140 | *EMPTY* |
|
||||
| 141 | *EMPTY* |
|
||||
| 142 | *EMPTY* |
|
||||
| 143 | *EMPTY* |
|
||||
|
||||
|
||||
## EMPTY KEYS
|
||||
|
||||
| Key |
|
||||
|------------|
|
||||
| 22 |
|
||||
| 38 - 39 |
|
||||
| 40 - 41 |
|
||||
| 44 - 45 |
|
||||
| 81 |
|
||||
| 136 - 137 |
|
||||
|
||||
|
||||
|
||||
## ROWS:
|
||||
|
||||
### ROW 0
|
||||
|
||||
| Index | Key |
|
||||
|--------|---------|
|
||||
| 0 | ESC |
|
||||
| 11 | F1 |
|
||||
| 17 | F2 |
|
||||
| 23 | F3 |
|
||||
| 28 | F4 |
|
||||
| 48 | F5 |
|
||||
| 53 | F6 |
|
||||
| 59 | F7 |
|
||||
| 65 | F8 |
|
||||
| 78 | F9 |
|
||||
| 84 | F10 |
|
||||
| 85 | F11 |
|
||||
| 86 | F12 |
|
||||
| 99 | PRINT |
|
||||
| 103 | SCROLL |
|
||||
| 108 | BREAK |
|
||||
|
||||
|
||||
### ROW 1
|
||||
|
||||
| Index | Key |
|
||||
|--------|------------|
|
||||
| 1 | ^ |
|
||||
| 6 | 1 |
|
||||
| 12 | 2 |
|
||||
| 18 | 3 |
|
||||
| 24 | 4 |
|
||||
| 29 | 5 |
|
||||
| 33 | 6 |
|
||||
| 49 | 7 |
|
||||
| 54 | 8 |
|
||||
| 60 | 9 |
|
||||
| 66 | 0 |
|
||||
| 72 | ß |
|
||||
| 79 | ´ |
|
||||
| 87 | BACKSPACE |
|
||||
| 100 | INS |
|
||||
| 104 | HOME |
|
||||
| 109 | PAGE UP |
|
||||
| 113 | NUM |
|
||||
| 119 | NUM / |
|
||||
| 124 | NUM * |
|
||||
| 129 | NUM - |
|
||||
|
||||
|
||||
### ROW 2
|
||||
|
||||
| Index | Key |
|
||||
|--------|------------|
|
||||
| 2 | TAB |
|
||||
| 7 | Q |
|
||||
| 13 | W |
|
||||
| 19 | E |
|
||||
| 25 | R |
|
||||
| 30 | T |
|
||||
| 34 | Z |
|
||||
| 50 | U |
|
||||
| 55 | I |
|
||||
| 61 | o |
|
||||
| 67 | P |
|
||||
| 73 | Ü |
|
||||
| 80 | + |
|
||||
| 101 | DEL |
|
||||
| 105 | END |
|
||||
| 110 | PAGE DOWN |
|
||||
| 114 | NUM 7 |
|
||||
| 120 | NUM 8 |
|
||||
| 125 | NUM 9 |
|
||||
|
||||
|
||||
### ROW 3
|
||||
|
||||
| Index | Key |
|
||||
|--------|---------|
|
||||
| 3 | CAPS |
|
||||
| 8 | A |
|
||||
| 14 | S |
|
||||
| 20 | D |
|
||||
| 26 | F |
|
||||
| 31 | G |
|
||||
| 35 | H |
|
||||
| 51 | J |
|
||||
| 56 | K |
|
||||
| 62 | L |
|
||||
| 68 | Ö |
|
||||
| 74 | Ä |
|
||||
| 88 | RETURN |
|
||||
| 115 | NUM 4 |
|
||||
| 121 | NUM 5 |
|
||||
| 126 | NUM 6 |
|
||||
| 130 | NUM + |
|
||||
|
||||
|
||||
### ROW 4
|
||||
|
||||
| Index | Key |
|
||||
|--------|-----------|
|
||||
| 4 | LSHIFT |
|
||||
| 9 | < |
|
||||
| 15 | Y |
|
||||
| 21 | X |
|
||||
| 27 | C |
|
||||
| 32 | V |
|
||||
| 36 | B |
|
||||
| 52 | N |
|
||||
| 57 | M |
|
||||
| 63 | ; |
|
||||
| 69 | : |
|
||||
| 75 | - |
|
||||
| 82 | RSHIFT |
|
||||
| 106 | ARROW UP |
|
||||
| 116 | NUM 1 |
|
||||
| 122 | NUM 2 |
|
||||
| 127 | NUM 3 |
|
||||
|
||||
|
||||
### ROW 5
|
||||
|
||||
| Index | Key |
|
||||
|--------|--------------|
|
||||
| 5 | CTRL |
|
||||
| 10 | L-SUPER |
|
||||
| 16 | L-ALT |
|
||||
| 37 | SPACE |
|
||||
| 70 | ALT+GR |
|
||||
| 76 | FN |
|
||||
| 83 | R-SUPER |
|
||||
| 89 | RCTRL |
|
||||
| 102 | ARROW LEFT |
|
||||
| 107 | ARROW DOWN |
|
||||
| 111 | ARROW RIGHT |
|
||||
| 117 | NUM 0 |
|
||||
| 128 | NUM , |
|
||||
| 131 | NUM RETURN |
|
||||
|
70
main.cpp
Normal file
70
main.cpp
Normal file
|
@ -0,0 +1,70 @@
|
|||
#include <unistd.h>
|
||||
#include <iostream>
|
||||
#include <VulcanoLE/API/HIDHelper.h>
|
||||
#include <VUtils/Logging.h>
|
||||
#include <csignal>
|
||||
#include <VulcanoLE/Audio/VisAudioRunner.h>
|
||||
#include <VUtils/Environment.h>
|
||||
|
||||
bool shouldRun = true;
|
||||
void my_handler(int s) {
|
||||
printf("Caught signal %d\n", s);
|
||||
shouldRun = false;
|
||||
}
|
||||
|
||||
int main(int argc, char **argv) {
|
||||
// signal handler!
|
||||
struct sigaction sigIntHandler;
|
||||
sigIntHandler.sa_handler = my_handler;
|
||||
sigemptyset(&sigIntHandler.sa_mask);
|
||||
sigIntHandler.sa_flags = 0;
|
||||
sigaction(SIGINT, &sigIntHandler, nullptr);
|
||||
// Load Config...
|
||||
VUtils::Environment config{VUtils::FileHandler::getFromHomeDir("/.config/VulcanoLE/state.env").c_str()};
|
||||
config.loadFile();
|
||||
LOG(R"(
|
||||
|
||||
[===============================================]
|
||||
_ _ _ _ _ _ _ _ _
|
||||
/ \ / \ / \ / \ / \ / \ / \ / \ / \
|
||||
( V | U | L | C | A | N | O ) ( L | E )
|
||||
\_/ \_/ \_/ \_/ \_/ \_/ \_/ \_/ \_/
|
||||
[===============================================]
|
||||
)")
|
||||
HIDHelper helper{};
|
||||
if (helper.openDevices() < 0) {
|
||||
ERR("Unable to find Keyboard!")
|
||||
exit(0);
|
||||
};
|
||||
usleep(10000);
|
||||
auto runner = VisAudioRunner::create();
|
||||
runner->env = &config;
|
||||
runner->plugins->init(&helper, runner->grabber);
|
||||
runner->init();
|
||||
runner->plugins->setCurrentMode(config.getAsInt("visual_mode", 1));
|
||||
while (shouldRun) {
|
||||
int mode;
|
||||
LOGWN("Enter Visual Mode: %d-%d < 0 = EXIT:\t", 0, 1)
|
||||
std::cin >> mode;
|
||||
if (std::cin.fail()) {
|
||||
ERR("ERROR -- You did not enter an integer")
|
||||
std::cin.clear();
|
||||
std::cin.ignore(std::numeric_limits<std::streamsize>::max(), '\n');
|
||||
continue;
|
||||
}
|
||||
if (mode < 0) {
|
||||
shouldRun = false;
|
||||
continue;
|
||||
}
|
||||
runner->plugins->setCurrentMode(mode);
|
||||
}
|
||||
DBG("Shutdown... Waiting for Thread to Finish!")
|
||||
runner->isActive = false;
|
||||
if (runner->thread.joinable()) {
|
||||
runner->thread.join();
|
||||
}
|
||||
DBG("Exit!")
|
||||
usleep(1000);
|
||||
config.saveFile();
|
||||
return 0;
|
||||
}
|
123
src/VUtils/Environment.cpp
Normal file
123
src/VUtils/Environment.cpp
Normal file
|
@ -0,0 +1,123 @@
|
|||
#include <VUtils/Environment.h>
|
||||
#include <VUtils/FileHandler.h>
|
||||
#include <VUtils/StringUtils.h>
|
||||
#include <VUtils/Logging.h>
|
||||
#include <sstream>
|
||||
#include <iomanip>
|
||||
|
||||
namespace VUtils {
|
||||
Environment::Environment(const char *filename) : m_filename(filename) {
|
||||
m_env[""] = "";
|
||||
}
|
||||
Environment::Environment() {
|
||||
m_env[""] = "";
|
||||
}
|
||||
|
||||
void Environment::loadFile() {
|
||||
DBG("Load ENV-File: %s", m_filename.c_str());
|
||||
if (!FileHandler::fileExists(m_filename)) {
|
||||
WARN("Cannot Load Env-File %s! File not found", m_filename.c_str());
|
||||
return;
|
||||
}
|
||||
auto content = StringUtils::trimCopy(FileHandler::readFile(m_filename));
|
||||
auto lines = StringUtils::split(content, "\n");
|
||||
for (auto &line : lines) {
|
||||
if (line.empty()) {
|
||||
continue;
|
||||
}
|
||||
auto split = StringUtils::split(line, "=");
|
||||
if (split.size() >= 2) {
|
||||
m_env[StringUtils::trimCopy(split[0])] = StringUtils::trimCopy(split[1]);
|
||||
} else {
|
||||
m_env[StringUtils::trimCopy(split[0])] = "true";
|
||||
}
|
||||
}
|
||||
DBG("Found %d Elements for Environment File %s", m_env.size(), m_filename.c_str());
|
||||
}
|
||||
|
||||
std::string &Environment::getEnv(const std::string &name, std::string def) {
|
||||
if (m_env.contains(name)) {
|
||||
return m_env[name];
|
||||
}
|
||||
auto *val = std::getenv(std::string(m_prefix + name).c_str());
|
||||
if (val) {
|
||||
m_env[name] = std::string(val);
|
||||
return m_env[name];
|
||||
}
|
||||
if (def.empty()) {
|
||||
return m_env[""];
|
||||
}
|
||||
m_env[name] = std::move(def);
|
||||
return m_env[name];
|
||||
}
|
||||
|
||||
bool Environment::hasEnv(const std::string &name) {
|
||||
return m_env.contains(name) || std::getenv(name.c_str()) != nullptr;
|
||||
}
|
||||
|
||||
int Environment::getAsInt(const std::string &name, int def) {
|
||||
return (int) getAsDouble(name, def);
|
||||
}
|
||||
|
||||
double Environment::getAsDouble(const std::string &name, double def) {
|
||||
char *end;
|
||||
auto *v = getEnv(name, "").c_str();
|
||||
double val = (int) std::strtod(v, &end);
|
||||
if (end == v) {
|
||||
setNumber(name.c_str(), def);
|
||||
return def;
|
||||
}
|
||||
return val;
|
||||
}
|
||||
|
||||
bool Environment::getAsBool(const std::string &name) {
|
||||
return getEnv(name, "false") == "true";
|
||||
}
|
||||
|
||||
void Environment::setPrefix(std::string prefix) {
|
||||
m_prefix = std::move(prefix);
|
||||
}
|
||||
|
||||
bool Environment::saveFile() {
|
||||
// override file if not exists!
|
||||
std::stringstream stream{};
|
||||
stream << std::setprecision(4);
|
||||
for (auto &element : m_env) {
|
||||
if (element.first.empty())
|
||||
continue;
|
||||
stream << element.first << "=" << element.second << "\n";
|
||||
}
|
||||
if (!FileHandler::createDirectoryIfNotExist(m_filename)) {
|
||||
ERR("Directory not exists or cannot create for: %s", m_filename.c_str());
|
||||
return false;
|
||||
}
|
||||
if (!FileHandler::writeFile(m_filename, stream.str())) {
|
||||
WARN("Cannot Save Env-File %s! Write failed", m_filename.c_str());
|
||||
return false;
|
||||
}
|
||||
DBG("Saved file to: %s", m_filename.c_str());
|
||||
return true;
|
||||
}
|
||||
|
||||
void Environment::setFile(const char *filename) {
|
||||
m_filename = filename;
|
||||
}
|
||||
|
||||
void Environment::set(const char *name, const char *value) {
|
||||
m_env[name] = value;
|
||||
}
|
||||
|
||||
// Small hack that set numbers to max precision
|
||||
void Environment::setNumber(const char *name, double value) {
|
||||
int newValue = (int) value;
|
||||
std::ostringstream out;
|
||||
|
||||
if (value != newValue) {
|
||||
out.precision(4);
|
||||
} else {
|
||||
out.precision(0);
|
||||
}
|
||||
out << std::fixed << value;
|
||||
m_env[name] = out.str();
|
||||
}
|
||||
}
|
91
src/VUtils/FileHandler.cpp
Normal file
91
src/VUtils/FileHandler.cpp
Normal file
|
@ -0,0 +1,91 @@
|
|||
#include <filesystem>
|
||||
#include <fstream>
|
||||
#include <vector>
|
||||
#include <fcntl.h>
|
||||
#include <sys/stat.h>
|
||||
#include <unistd.h>
|
||||
#include <VUtils/FileHandler.h>
|
||||
#include <VUtils/Logging.h>
|
||||
#include <pwd.h>
|
||||
|
||||
namespace VUtils {
|
||||
|
||||
bool FileHandler::fileExists(const std::string &fileName) {
|
||||
return std::filesystem::exists(fileName);
|
||||
}
|
||||
|
||||
std::string FileHandler::readFile(const std::string &fileName) {
|
||||
std::ifstream ifs(fileName.c_str(), std::ios::in | std::ios::binary | std::ios::ate);
|
||||
std::ifstream::pos_type fileSize = ifs.tellg();
|
||||
ifs.seekg(0, std::ios::beg);
|
||||
|
||||
std::vector<char> bytes(fileSize);
|
||||
ifs.read(bytes.data(), fileSize);
|
||||
|
||||
return std::string(bytes.data(), fileSize);
|
||||
}
|
||||
|
||||
bool FileHandler::writeFile(const std::string &fileName, const std::string &content) {
|
||||
try {
|
||||
std::ofstream ofs(fileName);
|
||||
ofs << content;
|
||||
ofs.close();
|
||||
} catch (std::exception &e) {
|
||||
ERR("Save Failed: %s", e.what());
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
bool FileHandler::isDirectory(const std::string &fileName) {
|
||||
return std::filesystem::is_directory(fileName);
|
||||
}
|
||||
|
||||
std::string FileHandler::getExtension(const std::string &fileName) {
|
||||
auto ext = std::filesystem::path(fileName).extension().string();
|
||||
if (ext.empty()) {
|
||||
return ext;
|
||||
}
|
||||
// remove first dot!
|
||||
return ext.erase(0, 1);
|
||||
}
|
||||
|
||||
int FileHandler::getFileID(const std::string &fileName) {
|
||||
return open(fileName.c_str(), O_RDONLY);
|
||||
}
|
||||
|
||||
long FileHandler::getFileSize(int fd) {
|
||||
struct stat stat_buf;
|
||||
int rc = fstat(fd, &stat_buf);
|
||||
return rc == 0 ? stat_buf.st_size : -1;
|
||||
}
|
||||
|
||||
void FileHandler::closeFD(int fd) {
|
||||
close(fd);
|
||||
}
|
||||
|
||||
std::string FileHandler::getFileName(const std::basic_string<char> &name) {
|
||||
auto p = std::filesystem::path(name);
|
||||
return p.filename().replace_extension("");
|
||||
}
|
||||
|
||||
bool FileHandler::createDirectoryIfNotExist(const std::basic_string<char> &fileName) {
|
||||
auto p = std::filesystem::path(fileName);
|
||||
if (!p.has_parent_path()) {
|
||||
return false;
|
||||
}
|
||||
if (FileHandler::isDirectory(p.parent_path())) {
|
||||
return true;
|
||||
}
|
||||
return std::filesystem::create_directories(p.parent_path());
|
||||
}
|
||||
char *FileHandler::getHomeDirectory() {
|
||||
char *homedir;
|
||||
if ((homedir = getenv("HOME")) == nullptr) {
|
||||
homedir = getpwuid(getuid())->pw_dir;
|
||||
}
|
||||
return homedir;
|
||||
}
|
||||
std::string FileHandler::getFromHomeDir(const std::basic_string<char> &path) {
|
||||
return std::string(getHomeDirectory() + path);
|
||||
}
|
||||
}
|
68
src/VUtils/Logging.cpp
Normal file
68
src/VUtils/Logging.cpp
Normal file
|
@ -0,0 +1,68 @@
|
|||
#include <VUtils/Logging.h>
|
||||
#include <cstdarg>
|
||||
#include <cstdio>
|
||||
|
||||
#ifndef ERROR_COLOR
|
||||
#define ERROR_COLOR 31
|
||||
#endif
|
||||
#ifndef INFO_COLOR
|
||||
#define INFO_COLOR 34
|
||||
#endif
|
||||
#ifndef DEBUG_COLOR
|
||||
#define DEBUG_COLOR 32
|
||||
#endif
|
||||
#ifndef WARN_COLOR
|
||||
#define WARN_COLOR 33
|
||||
#endif
|
||||
|
||||
void Logging::debug(bool newLine, const char *file, const char *func, ...) noexcept {
|
||||
va_list args;
|
||||
va_start(args, func);
|
||||
auto log = va_arg(args, const char*);
|
||||
vprintf(Logging::format(newLine, PrintType::DBG, log, file, func).c_str(), args);
|
||||
va_end(args);
|
||||
}
|
||||
|
||||
void Logging::log(bool newLine, const char *file, const char *func, ...) noexcept {
|
||||
va_list args;
|
||||
va_start(args, func);
|
||||
auto log = va_arg(args, const char*);
|
||||
vprintf(Logging::format(newLine, PrintType::LOG, log, file, func).c_str(), args);
|
||||
va_end(args);
|
||||
}
|
||||
|
||||
void Logging::error(bool newLine, const char *file, const char *func, ...) noexcept {
|
||||
va_list args;
|
||||
va_start(args, func);
|
||||
auto log = va_arg(args, const char*);
|
||||
vprintf(Logging::format(newLine, PrintType::ERROR, log, file, func).c_str(), args);
|
||||
va_end(args);
|
||||
}
|
||||
|
||||
void Logging::warn(bool newLine, const char *file, const char *func, ...) noexcept {
|
||||
va_list args;
|
||||
va_start(args, func);
|
||||
auto log = va_arg(args, const char*);
|
||||
vprintf(Logging::format(newLine, PrintType::WARN, log, file, func).c_str(), args);
|
||||
va_end(args);
|
||||
}
|
||||
|
||||
std::string Logging::format(bool newLine, PrintType type, const char *log, const char *file, const char *func) {
|
||||
auto pre = getPrefix(type, std::string(VUtils::FileHandler::getFileName(file) + "::" + func).c_str());
|
||||
pre += log;
|
||||
pre += "\033[0m";
|
||||
if (newLine) pre += "\n";
|
||||
return pre;
|
||||
}
|
||||
std::string Logging::getPrefix(PrintType type, const char *s) {
|
||||
switch (type) {
|
||||
case PrintType::ERROR:
|
||||
return "\033[1;" + std::to_string(ERROR_COLOR) + "m[ERROR][" + std::string(s) + "] >> ";
|
||||
case PrintType::DBG:
|
||||
return "\033[1;" + std::to_string(DEBUG_COLOR) + "m[DEBUG][" + std::string(s) + "] >> ";
|
||||
case PrintType::WARN:
|
||||
return "\033[1;" + std::to_string(WARN_COLOR) + "m[WARN][" + std::string(s) + "] >> ";
|
||||
default:
|
||||
return "\033[1;" + std::to_string(INFO_COLOR) + "m[LOG][" + std::string(s) + "] >> ";
|
||||
}
|
||||
}
|
39
src/VUtils/Pool.cpp
Normal file
39
src/VUtils/Pool.cpp
Normal file
|
@ -0,0 +1,39 @@
|
|||
#include <VUtils/Pool.h>
|
||||
#include <VUtils/Logging.h>
|
||||
|
||||
namespace VUtils {
|
||||
Pool::Pool(const char *name) : m_name(name) {
|
||||
}
|
||||
|
||||
Pool::~Pool() {
|
||||
delete[] m_threads;
|
||||
}
|
||||
|
||||
void Pool::setThreadCount(int count) {
|
||||
if (m_isCreated) return;
|
||||
m_count = count == -1 ? std::thread::hardware_concurrency() : count;
|
||||
}
|
||||
|
||||
void Pool::joinFirstThread() {
|
||||
if (!m_isCreated) return;
|
||||
if (m_threads[0].joinable()) {
|
||||
m_threads[0].join();
|
||||
}
|
||||
}
|
||||
|
||||
void Pool::create(PoolWorker &worker) {
|
||||
if (m_isCreated) return;
|
||||
m_isCreated = true;
|
||||
m_worker = &worker;
|
||||
m_threads = new std::thread[m_count];
|
||||
for (int i = 0; i < m_count; ++i) {
|
||||
m_threads[i] = std::thread(&Pool::execute, this);
|
||||
}
|
||||
DBG("Created %d Threads for ThreadPool %s", m_count, m_name)
|
||||
}
|
||||
|
||||
void Pool::execute() {
|
||||
m_worker->run();
|
||||
}
|
||||
}
|
||||
|
106
src/VUtils/StringUtils.cpp
Normal file
106
src/VUtils/StringUtils.cpp
Normal file
|
@ -0,0 +1,106 @@
|
|||
#include <VUtils/StringUtils.h>
|
||||
|
||||
#include <iomanip>
|
||||
#include <sstream>
|
||||
|
||||
namespace VUtils {
|
||||
void StringUtils::leftTrim(std::string &s) {
|
||||
s.erase(s.begin(), std::find_if(s.begin(), s.end(),
|
||||
std::not1(std::ptr_fun<int, int>(std::isspace))));
|
||||
}
|
||||
|
||||
void StringUtils::rightTrim(std::string &s) {
|
||||
s.erase(std::find_if(s.rbegin(), s.rend(),
|
||||
std::not1(std::ptr_fun<int, int>(std::isspace)))
|
||||
.base(),
|
||||
s.end());
|
||||
}
|
||||
|
||||
void StringUtils::trim(std::string &s) {
|
||||
leftTrim(s);
|
||||
rightTrim(s);
|
||||
}
|
||||
|
||||
std::string StringUtils::leftTrimCopy(std::string s) {
|
||||
leftTrim(s);
|
||||
return s;
|
||||
}
|
||||
|
||||
std::string StringUtils::rightTrimCopy(std::string s) {
|
||||
rightTrim(s);
|
||||
return s;
|
||||
}
|
||||
|
||||
std::string StringUtils::trimCopy(std::string s) {
|
||||
trim(s);
|
||||
return s;
|
||||
}
|
||||
|
||||
std::vector<std::string> StringUtils::split(const std::string &s, const std::string &delimiter) {
|
||||
size_t pos_start = 0, pos_end, delim_len = delimiter.length();
|
||||
std::string token;
|
||||
std::vector<std::string> res;
|
||||
|
||||
while ((pos_end = s.find(delimiter, pos_start)) != std::string::npos) {
|
||||
token = s.substr(pos_start, pos_end - pos_start);
|
||||
pos_start = pos_end + delim_len;
|
||||
res.push_back(token);
|
||||
}
|
||||
|
||||
res.push_back(s.substr(pos_start));
|
||||
return res;
|
||||
}
|
||||
|
||||
std::string StringUtils::urlDecode(const std::string &val) {
|
||||
std::string ret;
|
||||
char ch;
|
||||
int i, ii;
|
||||
for (i = 0; i < val.length(); i++) {
|
||||
if (int(val[ i ]) == 37) {
|
||||
sscanf(val.substr(i + 1, 2).c_str(), "%x", &ii);
|
||||
ch = static_cast<char>(ii);
|
||||
ret += ch;
|
||||
i = i + 2;
|
||||
} else {
|
||||
ret += val[ i ];
|
||||
}
|
||||
}
|
||||
return (ret);
|
||||
}
|
||||
|
||||
std::string StringUtils::urlEncode(const std::string &value) {
|
||||
std::ostringstream escaped;
|
||||
escaped.fill('0');
|
||||
escaped << std::hex;
|
||||
|
||||
for (char c : value) {
|
||||
if (isalnum(c) || c == '-' || c == '_' || c == '.' || c == '~') {
|
||||
escaped << c;
|
||||
continue;
|
||||
}
|
||||
escaped << std::uppercase;
|
||||
escaped << '%' << std::setw(2) << int((unsigned char) c);
|
||||
escaped << std::nouppercase;
|
||||
}
|
||||
|
||||
return escaped.str();
|
||||
}
|
||||
std::string StringUtils::join(const std::vector<std::string> &vector, const std::string &delimiter) {
|
||||
std::stringstream string;
|
||||
for (int i = 0; i < vector.size(); ++i) {
|
||||
if (i != 0)
|
||||
string << delimiter;
|
||||
string << vector[ i ];
|
||||
|
||||
}
|
||||
return string.str();
|
||||
}
|
||||
bool StringUtils::hasNullByte(int size, const char *string) {
|
||||
for (int i = 0; i < size; ++i) {
|
||||
if (string[ i ] == '\0') {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
}// namespace VUtils
|
164
src/VulcanoLE/API/HIDHelper.cpp
Normal file
164
src/VulcanoLE/API/HIDHelper.cpp
Normal file
|
@ -0,0 +1,164 @@
|
|||
#include <cstring>
|
||||
#include <cstdio>
|
||||
#include <fcntl.h>
|
||||
#include <execution>
|
||||
#include <sys/ioctl.h>
|
||||
|
||||
#include <VulcanoLE/API/HIDHelper.h>
|
||||
#include <VUtils/Logging.h>
|
||||
|
||||
#define RV_CTRL_INTERFACE 1
|
||||
#define RV_LED_INTERFACE 3
|
||||
#define RV_VENDOR 0x1e7d
|
||||
|
||||
|
||||
HIDHelper::HIDHelper() = default;
|
||||
HIDHelper::~HIDHelper() = default;
|
||||
int HIDHelper::openDevices() {
|
||||
int p = 0;
|
||||
while (m_products[ p ]) {
|
||||
unsigned short product_id = m_products[ p ];
|
||||
// For LED device, use hidapi-libusb, since we need to disconnect it
|
||||
// from the default kernel driver.
|
||||
struct hid_device_info *devs = nullptr;
|
||||
m_ledDevice = nullptr;
|
||||
devs = hid_enumerate(RV_VENDOR, product_id);
|
||||
if (!findLED(devs, product_id)) {
|
||||
if (devs) {
|
||||
hid_free_enumeration(devs);
|
||||
devs = nullptr;
|
||||
};
|
||||
if (m_ledDevice) hid_close(m_ledDevice);
|
||||
p++;
|
||||
continue;
|
||||
}
|
||||
|
||||
// For CTRL device, use native HIDRAW access. After sending the init
|
||||
// sequence, we will close it.
|
||||
|
||||
struct udev *udev = udev_new();
|
||||
struct udev_enumerate *enumerate = udev_enumerate_new(udev);
|
||||
udev_enumerate_add_match_subsystem(enumerate, "hidraw");
|
||||
udev_enumerate_scan_devices(enumerate);
|
||||
struct udev_list_entry *entries = udev_enumerate_get_list_entry(enumerate);
|
||||
struct udev_list_entry *cur;
|
||||
udev_list_entry_foreach(cur, entries) {
|
||||
struct udev_device *usb_dev = nullptr;
|
||||
struct udev_device *raw_dev = nullptr;
|
||||
if (ctrlDeviceWork(cur, usb_dev, raw_dev, udev, product_id)) {
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (ctrl_device < 1) {
|
||||
ERR("open_device(%04hx, %04hx): No CTRL device found", RV_VENDOR, product_id);
|
||||
if (devs) {
|
||||
hid_free_enumeration(devs);
|
||||
devs = nullptr;
|
||||
};
|
||||
if (m_ledDevice) hid_close(m_ledDevice);
|
||||
p++;
|
||||
continue;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
return -1;
|
||||
}
|
||||
bool HIDHelper::ctrlDeviceWork(
|
||||
udev_list_entry *cur,
|
||||
udev_device *usb_dev,
|
||||
udev_device *raw_dev,
|
||||
udev *udev,
|
||||
unsigned int product_id
|
||||
) {
|
||||
const char *sysfs_path = udev_list_entry_get_name(cur);
|
||||
if (!sysfs_path)
|
||||
return false;
|
||||
|
||||
raw_dev = udev_device_new_from_syspath(udev, sysfs_path);
|
||||
const char *dev_path = udev_device_get_devnode(raw_dev);
|
||||
if (!dev_path)
|
||||
return false;
|
||||
|
||||
usb_dev = udev_device_get_parent_with_subsystem_devtype(
|
||||
raw_dev,
|
||||
"usb",
|
||||
"usb_interface");
|
||||
if (!usb_dev)
|
||||
return false;
|
||||
|
||||
const char *info = udev_device_get_sysattr_value(usb_dev, "uevent");
|
||||
if (!info)
|
||||
return false;
|
||||
|
||||
const char *itf = udev_device_get_sysattr_value(usb_dev, "bInterfaceNumber");
|
||||
if (!itf)
|
||||
return false;
|
||||
|
||||
// We're looking for vid/pid and interface number
|
||||
if (atoi(itf) == RV_CTRL_INTERFACE) {
|
||||
char searchstr[64];
|
||||
snprintf(searchstr, 64, "PRODUCT=%hx/%hx", RV_VENDOR, product_id);
|
||||
|
||||
if (strstr(info, searchstr) != nullptr) {
|
||||
ctrl_device = open(dev_path, O_RDWR);
|
||||
if (!ctrl_device) {
|
||||
ERR("open_device(%04hx, %04hx): Unable to open CTRL device at %s", RV_VENDOR, product_id,
|
||||
dev_path);
|
||||
return false;
|
||||
}
|
||||
LOG("open_device(%04hx, %04hx): CTRL interface at %s", RV_VENDOR, product_id, dev_path);
|
||||
return true;
|
||||
}
|
||||
}
|
||||
if (raw_dev) udev_device_unref(raw_dev);
|
||||
return false;
|
||||
}
|
||||
|
||||
bool HIDHelper::findLED(hid_device_info *devs, unsigned int product_id) {
|
||||
struct hid_device_info *dev = nullptr;
|
||||
dev = devs;
|
||||
while (dev) {
|
||||
if (dev->interface_number == RV_LED_INTERFACE) {
|
||||
LOG("open_device(%04hx, %04hx): LED interface at USB path %s", RV_VENDOR, product_id, dev->path);
|
||||
m_ledDevice = hid_open_path(dev->path);
|
||||
if (!m_ledDevice || hid_set_nonblocking(m_ledDevice, 1) < 0) {
|
||||
ERR("open_device(%04hx, %04hx): Unable to open LED interface %s", RV_VENDOR, product_id,
|
||||
dev->path);
|
||||
return false;
|
||||
}
|
||||
} else {
|
||||
DBG("open_device(%04hx, %04hx): ignoring non-LED interface #%d", RV_VENDOR, product_id,
|
||||
dev->interface_number);
|
||||
}
|
||||
dev = dev->next;
|
||||
}
|
||||
|
||||
if (!m_ledDevice) {
|
||||
DBG("open_device(%04hx, %04hx): No LED device found", RV_VENDOR, product_id);
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
int HIDHelper::get_feature_report(unsigned char *buf, int size) const {
|
||||
int res = ioctl(ctrl_device, HIDIOCGFEATURE(size), buf);
|
||||
if (res < 0) {
|
||||
return 0;
|
||||
}
|
||||
return size;
|
||||
}
|
||||
|
||||
int HIDHelper::send_feature_report(unsigned char *buf, int size) const {
|
||||
int res = ioctl(ctrl_device, HIDIOCSFEATURE(size), buf);
|
||||
if (res < 0) {
|
||||
return 0;
|
||||
}
|
||||
return size;
|
||||
}
|
||||
|
||||
void HIDHelper::close_ctrl_device() {
|
||||
if (ctrl_device) close(ctrl_device);
|
||||
ctrl_device = 0;
|
||||
}
|
194
src/VulcanoLE/Audio/AudioGrabber.cpp
Normal file
194
src/VulcanoLE/Audio/AudioGrabber.cpp
Normal file
|
@ -0,0 +1,194 @@
|
|||
#include <cstring>
|
||||
#include <cmath>
|
||||
#include <VulcanoLE/Audio/AudioGrabber.h>
|
||||
#include <VUtils/Logging.h>
|
||||
|
||||
|
||||
AudioGrabber::AudioGrabber() = default;
|
||||
AudioGrabber::~AudioGrabber() = default;
|
||||
|
||||
bool AudioGrabber::read(pcm_stereo_sample *buffer, uint32_t buffer_size) {
|
||||
auto buffer_size_bytes = static_cast<size_t>(sizeof(pcm_stereo_sample) * buffer_size);
|
||||
if (m_pulseaudio_simple == nullptr) {
|
||||
open_pulseaudio_source(static_cast<uint32_t>(buffer_size_bytes));
|
||||
}
|
||||
|
||||
if (m_pulseaudio_simple != nullptr) {
|
||||
memset(buffer, 0, buffer_size_bytes);
|
||||
int32_t error_code;
|
||||
auto return_code = pa_simple_read(m_pulseaudio_simple, buffer,
|
||||
buffer_size_bytes, &error_code);
|
||||
if (return_code < 0) {
|
||||
WARN("Could not finish reading pulse Audio stream buffer\n bytes read: %d buffer\n size: %d", return_code,
|
||||
buffer_size_bytes)
|
||||
// zero out buffer
|
||||
memset(buffer, 0, buffer_size_bytes);
|
||||
pa_simple_free(m_pulseaudio_simple);
|
||||
m_pulseaudio_simple = nullptr;
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
// Success fully read entire buffer
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
|
||||
void AudioGrabber::populate_default_source_name() {
|
||||
pa_mainloop_api *mainloop_api;
|
||||
pa_context *pulseaudio_context;
|
||||
m_pulseaudio_mainloop = pa_mainloop_new();
|
||||
mainloop_api = pa_mainloop_get_api(m_pulseaudio_mainloop);
|
||||
pulseaudio_context = pa_context_new(mainloop_api, "VulcanoLE device list");
|
||||
pa_context_connect(pulseaudio_context, nullptr, PA_CONTEXT_NOFLAGS,
|
||||
nullptr);
|
||||
pa_context_set_state_callback(pulseaudio_context,
|
||||
pulseaudio_context_state_callback,
|
||||
reinterpret_cast<void *>(this));
|
||||
|
||||
int ret;
|
||||
if (pa_mainloop_run(m_pulseaudio_mainloop, &ret) < 0) {
|
||||
ERR("Could not open pulseaudio mainloop to find default device name: %d", ret)
|
||||
}
|
||||
}
|
||||
|
||||
bool AudioGrabber::open_pulseaudio_source(uint32_t max_buffer_size) {
|
||||
int32_t error_code = 0;
|
||||
static const pa_sample_spec sample_spec = { PA_SAMPLE_FLOAT32NE, k_sample_rate,
|
||||
k_channels };
|
||||
static const pa_buffer_attr buffer_attr = { max_buffer_size, 0, 0, 0,
|
||||
(max_buffer_size / 2) };
|
||||
populate_default_source_name();
|
||||
if (!m_pulseaudio_default_source_name.empty()) {
|
||||
m_pulseaudio_simple =
|
||||
pa_simple_new(nullptr, k_record_stream_name, PA_STREAM_RECORD,
|
||||
m_pulseaudio_default_source_name.c_str(),
|
||||
k_record_stream_description, &sample_spec,
|
||||
nullptr, &buffer_attr, &error_code);
|
||||
}
|
||||
if (m_pulseaudio_simple == nullptr) {
|
||||
m_pulseaudio_simple =
|
||||
pa_simple_new(nullptr, k_record_stream_name, PA_STREAM_RECORD,
|
||||
nullptr, k_record_stream_description,
|
||||
&sample_spec, nullptr, &buffer_attr, &error_code);
|
||||
}
|
||||
if (m_pulseaudio_simple == nullptr) {
|
||||
m_pulseaudio_simple =
|
||||
pa_simple_new(nullptr, k_record_stream_name, PA_STREAM_RECORD,
|
||||
"0", k_record_stream_description, &sample_spec,
|
||||
nullptr, &buffer_attr, &error_code);
|
||||
}
|
||||
if (m_pulseaudio_simple != nullptr) {
|
||||
return true;
|
||||
}
|
||||
|
||||
ERR("Could not open pulseaudio source: %s", pa_strerror(error_code))
|
||||
return false;
|
||||
}
|
||||
|
||||
void AudioGrabber::pulseaudio_context_state_callback(pa_context *c, void *userdata) {
|
||||
switch (pa_context_get_state(c)) {
|
||||
case PA_CONTEXT_UNCONNECTED:
|
||||
case PA_CONTEXT_CONNECTING:
|
||||
case PA_CONTEXT_AUTHORIZING:
|
||||
case PA_CONTEXT_SETTING_NAME:
|
||||
break;
|
||||
|
||||
case PA_CONTEXT_READY: {
|
||||
pa_operation_unref(pa_context_get_server_info(
|
||||
c, pulseaudio_server_info_callback, userdata));
|
||||
break;
|
||||
}
|
||||
|
||||
case PA_CONTEXT_FAILED:
|
||||
case PA_CONTEXT_TERMINATED:
|
||||
auto *src =
|
||||
reinterpret_cast<AudioGrabber *>(userdata);
|
||||
pa_mainloop_quit(src->m_pulseaudio_mainloop, 0);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
void AudioGrabber::pulseaudio_server_info_callback(pa_context *context, const pa_server_info *i, void *userdata) {
|
||||
if (i != nullptr) {
|
||||
auto *src = reinterpret_cast<AudioGrabber *>(userdata);
|
||||
std::string name = i->default_sink_name;
|
||||
name.append(k_default_monitor_postfix);
|
||||
|
||||
src->m_pulseaudio_default_source_name = name;
|
||||
|
||||
// stop mainloop after finding default name
|
||||
pa_mainloop_quit(src->m_pulseaudio_mainloop, 0);
|
||||
}
|
||||
}
|
||||
|
||||
AudioGrabber *AudioGrabber::createAudioGrabber() {
|
||||
auto *grabber = new AudioGrabber();
|
||||
return grabber;
|
||||
}
|
||||
|
||||
void AudioGrabber::init() {
|
||||
m_pcm_buffer = static_cast<pcm_stereo_sample *>(calloc(BUFFER_SIZE, sizeof(pcm_stereo_sample)));
|
||||
if (env != nullptr)
|
||||
m_scale = env->getAsDouble("audio_scale", 1.0);
|
||||
DBG("SET Audio Scale: %.3f", m_scale)
|
||||
loudness = 0.0;
|
||||
}
|
||||
|
||||
void AudioGrabber::calculateRMS(pcm_stereo_sample *pFrame) {
|
||||
float square = 0, mean;
|
||||
for (int i = 0; i < BUFFER_SIZE; i++) {
|
||||
float left = pFrame[0].l;
|
||||
square += std::pow(left, 2);
|
||||
}
|
||||
mean = (square / (float) (BUFFER_SIZE));
|
||||
loudness = std::sqrt(mean);
|
||||
}
|
||||
|
||||
void AudioGrabber::calculatePEAK(pcm_stereo_sample *pFrame) {
|
||||
float max = 0;
|
||||
for (int i = 0; i < BUFFER_SIZE; i++) {
|
||||
float left = std::abs(pFrame[0].l);
|
||||
if (left > max) {
|
||||
max = left;
|
||||
}
|
||||
}
|
||||
loudness = max;
|
||||
}
|
||||
|
||||
float AudioGrabber::getLoudness() {
|
||||
std::unique_lock<std::mutex> lck(m_mtx);
|
||||
return (float) loudness;
|
||||
}
|
||||
|
||||
bool AudioGrabber::doWork() {
|
||||
std::unique_lock<std::mutex> lck(m_mtx);
|
||||
if (this->read(m_pcm_buffer, BUFFER_SIZE)) {
|
||||
for (int i = 0; i < BUFFER_SIZE; ++i) {
|
||||
// my system is fucking quite
|
||||
m_pcm_buffer[i].l *= m_scale;
|
||||
m_pcm_buffer[i].r *= m_scale;
|
||||
}
|
||||
switch (requestMode) {
|
||||
case ReqMode::FFT:
|
||||
fft.process(m_pcm_buffer);
|
||||
break;
|
||||
case ReqMode::RMS:
|
||||
calculateRMS(m_pcm_buffer);
|
||||
break;
|
||||
case ReqMode::PEAK:
|
||||
calculatePEAK(m_pcm_buffer);
|
||||
break;
|
||||
default:
|
||||
fft.process(m_pcm_buffer);
|
||||
calculateRMS(m_pcm_buffer);
|
||||
calculatePEAK(m_pcm_buffer);
|
||||
}
|
||||
return true;
|
||||
} else {
|
||||
DBG("Wait for Data")
|
||||
return false;
|
||||
}
|
||||
}
|
55
src/VulcanoLE/Audio/FFT.cpp
Normal file
55
src/VulcanoLE/Audio/FFT.cpp
Normal file
|
@ -0,0 +1,55 @@
|
|||
#include <VulcanoLE/Audio/FFT.h>
|
||||
#include <VUtils/Logging.h>
|
||||
|
||||
void FFT::process(pcm_stereo_sample *pFrame) {
|
||||
std::unique_lock<std::mutex> lck(m_mtx);
|
||||
prepareInput(pFrame, FFT_SIZE);
|
||||
m_fftw_plan_left = fftw_plan_dft_r2c_1d(static_cast<int>(BUFFER_SIZE), m_fftw_input_left,
|
||||
m_fftw_output_left, FFTW_ESTIMATE);
|
||||
m_fftw_plan_right = fftw_plan_dft_r2c_1d(static_cast<int>(BUFFER_SIZE), m_fftw_input_right,
|
||||
m_fftw_output_right, FFTW_ESTIMATE);
|
||||
fftw_execute(m_fftw_plan_left);
|
||||
fftw_execute(m_fftw_plan_right);
|
||||
fftw_destroy_plan(m_fftw_plan_left);
|
||||
fftw_destroy_plan(m_fftw_plan_right);
|
||||
for (int i = 0; i < FFT_SIZE; ++i) {
|
||||
double left = (double(*m_fftw_output_left[i]));
|
||||
double right = (double(*m_fftw_output_right[i]));
|
||||
m_sample->leftChannel[i] = left;
|
||||
m_sample->rightChannel[i] = right;
|
||||
}
|
||||
}
|
||||
|
||||
// return vector of floats!
|
||||
outputSample *FFT::getData() {
|
||||
std::unique_lock<std::mutex> lck(m_mtx);
|
||||
return m_sample;
|
||||
}
|
||||
|
||||
bool FFT::prepareInput(pcm_stereo_sample *buffer, uint32_t sample_size) {
|
||||
bool is_silent = true;
|
||||
for (auto i = 0u; i < sample_size; ++i) {
|
||||
m_fftw_input_left[i] = buffer[i].l;
|
||||
m_fftw_input_right[i] = buffer[i].r;
|
||||
if (is_silent && (m_fftw_input_left[i] > 0 || m_fftw_input_right[i] > 0)) is_silent = false;
|
||||
}
|
||||
return is_silent;
|
||||
}
|
||||
|
||||
FFT::FFT() {
|
||||
m_fftw_results = (static_cast<size_t>(BUFFER_SIZE) / 2) + 1;
|
||||
|
||||
m_fftw_input_left = static_cast<double *>(
|
||||
fftw_malloc(sizeof(double) * BUFFER_SIZE));
|
||||
m_fftw_input_right = static_cast<double *>(
|
||||
fftw_malloc(sizeof(double) * BUFFER_SIZE));
|
||||
|
||||
m_fftw_output_left = static_cast<fftw_complex *>(
|
||||
fftw_malloc(sizeof(fftw_complex) * m_fftw_results));
|
||||
m_fftw_output_right = static_cast<fftw_complex *>(
|
||||
fftw_malloc(sizeof(fftw_complex) * m_fftw_results));
|
||||
}
|
||||
FFT::~FFT() {
|
||||
delete m_sample;
|
||||
}
|
||||
|
37
src/VulcanoLE/Audio/VisAudioRunner.cpp
Normal file
37
src/VulcanoLE/Audio/VisAudioRunner.cpp
Normal file
|
@ -0,0 +1,37 @@
|
|||
#include <VulcanoLE/Audio/VisAudioRunner.h>
|
||||
#include <VUtils/Logging.h>
|
||||
|
||||
VisAudioRunner::VisAudioRunner(AudioGrabber *ag, VIZ::VisPlugins *vp) : grabber(ag), plugins(vp) {}
|
||||
|
||||
VisAudioRunner::~VisAudioRunner() {
|
||||
delete plugins;
|
||||
delete grabber;
|
||||
}
|
||||
|
||||
VisAudioRunner *VisAudioRunner::create() {
|
||||
return new VisAudioRunner(AudioGrabber::createAudioGrabber(), new VIZ::VisPlugins());
|
||||
}
|
||||
|
||||
|
||||
void VisAudioRunner::init() {
|
||||
grabber->env = env;
|
||||
plugins->env = env;
|
||||
grabber->init();
|
||||
plugins->on_startup();
|
||||
LOG("Create Visual Audio Runner Thread")
|
||||
thread = std::thread(&VisAudioRunner::run, this);
|
||||
}
|
||||
|
||||
void VisAudioRunner::run() const {
|
||||
usleep(5000);
|
||||
while (isActive) {
|
||||
if (grabber->doWork()) {
|
||||
plugins->on_tick();
|
||||
} else {
|
||||
std::this_thread::sleep_for(std::chrono::milliseconds(100ul));
|
||||
}
|
||||
}
|
||||
usleep(50000);
|
||||
DBG("SHUTDOWN HOOk")
|
||||
plugins->on_shutdown();
|
||||
}
|
242
src/VulcanoLE/Keyboards/Vulcan121.cpp
Normal file
242
src/VulcanoLE/Keyboards/Vulcan121.cpp
Normal file
|
@ -0,0 +1,242 @@
|
|||
#include <cstring>
|
||||
#include <execution>
|
||||
#include <VulcanoLE/Keyboards/Vulcan121.h>
|
||||
#include <VUtils/Logging.h>
|
||||
|
||||
Vulcan121::Vulcan121(HIDHelper *helper)
|
||||
: m_helper(helper) {}
|
||||
|
||||
int Vulcan121::send_led_map(led_map *src, bool deleteMap) {
|
||||
int i, k;
|
||||
rgba rgb;
|
||||
unsigned char hwmap[444]{};
|
||||
unsigned char workbuf[65];
|
||||
memset(hwmap, 0, sizeof(hwmap));
|
||||
for (k = 0; k < NUM_KEYS; k++) {
|
||||
rgb = rv_fixed[ k ] ? *(rv_fixed[ k ]) : (src ? src->key[ k ] : rv_color_off);
|
||||
|
||||
rgb.r = (rgb.r > 255) ? 255 : (rgb.r < 0) ? 0 : rgb.r;
|
||||
rgb.g = (rgb.g > 255) ? 255 : (rgb.g < 0) ? 0 : rgb.g;
|
||||
rgb.b = (rgb.b > 255) ? 255 : (rgb.b < 0) ? 0 : rgb.b;
|
||||
rgb.a = (rgb.a > 255) ? 255 : (rgb.a < 0) ? 0 : rgb.a;
|
||||
double factor = rgb.a / 255.0;
|
||||
rgb.r *= factor;
|
||||
rgb.g *= factor;
|
||||
rgb.b *= factor;
|
||||
|
||||
int offset = ((k / 12) * 36) + (k % 12);
|
||||
hwmap[ offset + 0 ] = (unsigned char) rgb.r;
|
||||
hwmap[ offset + 12 ] = (unsigned char) rgb.g;
|
||||
hwmap[ offset + 24 ] = (unsigned char) rgb.b;
|
||||
}
|
||||
|
||||
// First chunk comes with header
|
||||
workbuf[ 0 ] = 0x00;
|
||||
workbuf[ 1 ] = 0xa1;
|
||||
workbuf[ 2 ] = 0x01;
|
||||
workbuf[ 3 ] = 0x01;
|
||||
workbuf[ 4 ] = 0xb4;
|
||||
memcpy(&workbuf[ 5 ], hwmap, 60);
|
||||
if (hid_write(m_helper->m_ledDevice, workbuf, 65) != 65) {
|
||||
if (deleteMap) {
|
||||
delete src;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
// Six more chunks
|
||||
for (i = 1; i < 7; i++) {
|
||||
workbuf[ 0 ] = 0x00;
|
||||
memcpy(&workbuf[ 1 ], &hwmap[ (i * 64) - 4 ], 64);
|
||||
if (hid_write(m_helper->m_ledDevice, workbuf, 65) != 65) {
|
||||
if (deleteMap) {
|
||||
delete src;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
if (deleteMap) {
|
||||
delete src;
|
||||
}
|
||||
return 1;
|
||||
}
|
||||
int Vulcan121::send_led_to(rgba rgb) {
|
||||
auto *map = new led_map();
|
||||
for (auto &i : map->key) {
|
||||
i = rgb;
|
||||
}
|
||||
int st = send_led_map(map, true);
|
||||
return st;
|
||||
}
|
||||
|
||||
led_map *Vulcan121::createEmptyLEDMap() {
|
||||
return new led_map();
|
||||
}
|
||||
|
||||
bool Vulcan121::send_init_sequence() {
|
||||
LOG("Sending device init sequence...")
|
||||
unsigned char a[9] = { 0x15, 0x05, 0x07, 0x0a, 0x0b, 0x06, 0x09, 0x0d, 0x13 };
|
||||
if (!query_ctrl_report(0x0f))
|
||||
return false;
|
||||
for (auto i : a) {
|
||||
if (!send_ctrl_report(i) || !wait_for_ctrl_dev()) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
m_helper->close_ctrl_device();
|
||||
return true;
|
||||
}
|
||||
|
||||
bool Vulcan121::query_ctrl_report(unsigned char id) {
|
||||
if (id != 0x0f) return false;
|
||||
unsigned char buffer[8] = {};
|
||||
int length = 8;
|
||||
buffer[ 0 ] = id;
|
||||
int res = m_helper->get_feature_report(buffer, length);
|
||||
if (res) {
|
||||
return true;
|
||||
}
|
||||
ERR("query_ctrl_report(%02hhx) failed", id);
|
||||
return false;
|
||||
}
|
||||
|
||||
bool Vulcan121::send_ctrl_report(unsigned char id) {
|
||||
unsigned char *buffer;
|
||||
int length;
|
||||
|
||||
switch (id) {
|
||||
case 0x15:
|
||||
buffer = (unsigned char *) "\x15\x00\x01";
|
||||
length = 3;
|
||||
break;
|
||||
case 0x05:
|
||||
buffer = (unsigned char *) "\x05\x04\x00\x04";
|
||||
length = 4;
|
||||
break;
|
||||
case 0x07:
|
||||
buffer = (unsigned char *) "\x07\x5f\x00\x3a\x00\x00\x3b\x00\x00\x3c\x00\x00\x3d\x00\x00\x3e" \
|
||||
"\x00\x00\x3f\x00\x00\x40\x00\x00\x41\x00\x00\x42\x00\x00\x43\x00" \
|
||||
"\x00\x44\x00\x00\x45\x00\x00\x46\x00\x00\x47\x00\x00\x48\x00\x00" \
|
||||
"\xb3\x00\x00\xb4\x00\x00\xb5\x00\x00\xb6\x00\x00\xc2\x00\x00\xc3" \
|
||||
"\x00\x00\xc0\x00\x00\xc1\x00\x00\xce\x00\x00\xcf\x00\x00\xcc\x00" \
|
||||
"\x00\xcd\x00\x00\x46\x00\x00\xfc\x00\x00\x48\x00\x00\xcd\x0e";
|
||||
length = 95;
|
||||
break;
|
||||
case 0x0a:
|
||||
buffer = (unsigned char *) "\x0a\x08\x00\xff\xf1\x00\x02\x02";
|
||||
length = 8;
|
||||
break;
|
||||
case 0x0b:
|
||||
buffer = (unsigned char *) "\x0b\x41\x00\x1e\x00\x00\x1f\x00\x00\x20\x00\x00\x21\x00\x00\x22" \
|
||||
"\x00\x00\x14\x00\x00\x1a\x00\x00\x08\x00\x00\x15\x00\x00\x17\x00" \
|
||||
"\x00\x04\x00\x00\x16\x00\x00\x07\x00\x00\x09\x00\x00\x0a\x00\x00" \
|
||||
"\x1d\x00\x00\x1b\x00\x00\x06\x00\x00\x19\x00\x00\x05\x00\x00\xde\x01";
|
||||
length = 65;
|
||||
break;
|
||||
case 0x06:
|
||||
buffer = (unsigned char *) "\x06\x85\x00\x3a\x29\x35\x1e\x2b\x39\xe1\xe0\x3b\x1f\x14\x1a\x04" \
|
||||
"\x64\x00\x00\x3d\x3c\x20\x21\x08\x16\x1d\xe2\x3e\x23\x22\x15\x07" \
|
||||
"\x1b\x06\x8b\x3f\x24\x00\x17\x0a\x09\x19\x91\x40\x41\x00\x1c\x18" \
|
||||
"\x0b\x05\x2c\x42\x26\x25\x0c\x0d\x0e\x10\x11\x43\x2a\x27\x2d\x12" \
|
||||
"\x0f\x36\x8a\x44\x45\x89\x2e\x13\x33\x37\x90\x46\x49\x4c\x2f\x30" \
|
||||
"\x34\x38\x88\x47\x4a\x4d\x31\x32\x00\x87\xe6\x48\x4b\x4e\x28\x52" \
|
||||
"\x50\xe5\xe7\xd2\x53\x5f\x5c\x59\x51\x00\xf1\xd1\x54\x60\x5d\x5a" \
|
||||
"\x4f\x8e\x65\xd0\x55\x61\x5e\x5b\x62\xa4\xe4\xfc\x56\x57\x85\x58" \
|
||||
"\x63\x00\x00\xc2\x24";
|
||||
length = 133;
|
||||
break;
|
||||
case 0x09:
|
||||
buffer = (unsigned char *) "\x09\x2b\x00\x49\x00\x00\x4a\x00\x00\x4b\x00\x00\x4c\x00\x00\x4d" \
|
||||
"\x00\x00\x4e\x00\x00\xa4\x00\x00\x8e\x00\x00\xd0\x00\x00\xd1\x00" \
|
||||
"\x00\x00\x00\x00\x01\x00\x00\x00\x00\xcd\x04";
|
||||
length = 43;
|
||||
break;
|
||||
case 0x0d:
|
||||
length = 443;
|
||||
buffer = (unsigned char *) "\x0d\xbb\x01\x00\x06\x0b\x05\x45\x83\xca\xca\xca\xca\xca\xca\xce" \
|
||||
"\xce\xd2\xce\xce\xd2\x19\x19\x19\x19\x19\x19\x23\x23\x2d\x23\x23" \
|
||||
"\x2d\xe0\xe0\xe0\xe0\xe0\xe0\xe3\xe3\xe6\xe3\xe3\xe6\xd2\xd2\xd5" \
|
||||
"\xd2\xd2\xd5\xd5\xd5\xd9\xd5\x00\xd9\x2d\x2d\x36\x2d\x2d\x36\x36" \
|
||||
"\x36\x40\x36\x00\x40\xe6\xe6\xe9\xe6\xe6\xe9\xe9\xe9\xec\xe9\x00" \
|
||||
"\xec\xd9\xd9\xdd\xd9\xdd\xdd\xe0\xe0\xdd\xe0\xe4\xe4\x40\x40\x4a" \
|
||||
"\x40\x4a\x4a\x53\x53\x4a\x53\x5d\x5d\xec\xec\xef\xec\xef\xef\xf2" \
|
||||
"\xf2\xef\xf2\xf5\xf5\xe4\xe4\x00\x00\x00\x00\x00\x00\x00\x00\x00" \
|
||||
"\x00\x5d\x5d\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\xf5\xf5\x00" \
|
||||
"\x00\x00\x00\x00\x00\x00\x00\x00\x00\xe4\xe4\xe8\xe8\xe8\xe8\xe8" \
|
||||
"\xeb\xeb\xeb\x00\xeb\x5d\x5d\x67\x67\x67\x67\x67\x70\x70\x70\x00" \
|
||||
"\x70\xf5\xf5\xf8\xf8\xf8\xf8\xf8\xfb\xfb\xfb\x00\xfb\xeb\xef\xef" \
|
||||
"\xef\x00\xef\xf0\xf0\xed\xf0\xf0\x00\x70\x7a\x7a\x7a\x00\x7a\x7a" \
|
||||
"\x7a\x6f\x7a\x7a\x00\xfb\xfd\xfd\xfd\x00\xfd\xf8\xf8\xea\xf8\xf8" \
|
||||
"\x00\xed\xed\xea\xed\xed\x00\xed\xea\xea\xf6\xe7\xea\x6f\x6f\x65" \
|
||||
"\x6f\x6f\x00\x6f\x65\x65\x66\x5a\x65\xea\xea\xdc\xea\xea\x00\xea" \
|
||||
"\xdc\xdc\x00\xce\xdc\xea\xe7\xe5\xe7\xe5\xe5\x00\x00\x00\x00\x00" \
|
||||
"\x00\x65\x5a\x50\x5a\x50\x50\x00\x00\x00\x00\x00\x00\xdc\xce\xc0" \
|
||||
"\xce\xc0\xc0\x00\x00\x00\x00\x00\x00\xe7\x00\x00\xe2\xe2\xe2\xe2" \
|
||||
"\xdf\xdf\xdf\xdf\xdf\x5a\x00\x00\x45\x45\x45\x45\x3b\x3b\x3b\x3b" \
|
||||
"\x3b\xce\x00\x00\xb2\xb2\xb2\xb2\xa4\xa4\xa4\xa4\xa4\xdc\xdc\xdc" \
|
||||
"\xdc\x00\xda\xda\xda\xda\xda\x00\xd7\x30\x30\x30\x30\x00\x26\x26" \
|
||||
"\x26\x26\x26\x00\x1c\x96\x96\x96\x96\x00\x88\x88\x88\x88\x88\x00" \
|
||||
"\x7a\xd7\xd7\xd7\x00\xd4\xd4\xd4\xd4\xd4\xd1\xd1\xd1\x1c\x1c\x1c" \
|
||||
"\x00\x11\x11\x11\x11\x11\x06\x06\x06\x7a\x7a\x7a\x00\x6c\x6c\x6c" \
|
||||
"\x6c\x6c\x5e\x5e\x5e\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00" \
|
||||
"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00" \
|
||||
"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x24\xcf";
|
||||
break;
|
||||
case 0x13:
|
||||
buffer = (unsigned char *) "\x13\x08\x01\x00\x00\x00\x00\x00";
|
||||
length = 8;
|
||||
break;
|
||||
default: ERR("UNKNOWN BUFFER OPTION")
|
||||
return false;
|
||||
}
|
||||
int res = m_helper->send_feature_report(buffer, length);
|
||||
if (res == length) return true;
|
||||
ERR("send_ctrl_report(%02hhx) failed", id)
|
||||
return false;
|
||||
}
|
||||
|
||||
bool Vulcan121::wait_for_ctrl_dev() {
|
||||
unsigned char buffer[] = { 0x04, 0x00, 0x00, 0x00 };
|
||||
int res;
|
||||
while (true) {
|
||||
// 150ms is the magic number here, should suffice on first try.
|
||||
usleep(10000);
|
||||
res = m_helper->get_feature_report(buffer, sizeof(buffer));
|
||||
if (res) {
|
||||
if (buffer[ 1 ] == 0x01) break;
|
||||
} else {
|
||||
ERR("rv_wait_for_ctrl_device() failed");
|
||||
return false;
|
||||
}
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
int Vulcan121::getColumnsForRow(int row) {
|
||||
if (row > 5) {
|
||||
WARN("Try to Access out of Bound %d max %d", row, 5)
|
||||
return 0;
|
||||
}
|
||||
return keyMapRow[row]->count;
|
||||
}
|
||||
|
||||
// @Todo Add Columngs
|
||||
int Vulcan121::getRowsForColumns(int col) {
|
||||
if (col > 5) {
|
||||
WARN("Try to Access out of Bound %d max %d", col, 5)
|
||||
return 0;
|
||||
}
|
||||
return keyMapCols[col]->count;
|
||||
}
|
||||
|
||||
int Vulcan121::getIndex(int row, int col) {
|
||||
if (row > 5) {
|
||||
WARN("Try to Access out of Bound %d max %d", row, 5)
|
||||
return 0;
|
||||
}
|
||||
if (col > keyMapRow[ row ]->count) {
|
||||
WARN("Try to Access out of Bound %d max %d", col, keyMapRow[row]->count)
|
||||
return 0;
|
||||
}
|
||||
return keyMapRow[row]->index[col];
|
||||
}
|
33
src/VulcanoLE/Scripts/Loudness.cpp
Normal file
33
src/VulcanoLE/Scripts/Loudness.cpp
Normal file
|
@ -0,0 +1,33 @@
|
|||
#include <VulcanoLE/Scripts/Loudness.h>
|
||||
|
||||
namespace VIZ {
|
||||
Loudness::Loudness(AudioGrabber *pGrabber, Vulcan121 *pVulcan121) : VIZ(pGrabber, pVulcan121) {}
|
||||
|
||||
void Loudness::on_setup() {
|
||||
keyboard->send_led_to({ 0, 0, 0, 0 });
|
||||
grabber->requestMode = AudioGrabber::ReqMode::PEAK;
|
||||
usleep(100000);
|
||||
}
|
||||
|
||||
void Loudness::on_tick() {
|
||||
auto data = Vulcan121::createEmptyLEDMap();
|
||||
float val = grabber->getLoudness();
|
||||
val = val > 1.0f ? 1.0f : val;
|
||||
double newVal = (val + lastVal) * 0.5;
|
||||
int maxCol = newVal * 24;
|
||||
for (int col = 0; col < maxCol; ++col) {
|
||||
for (int i = 0; i < keyboardData.num_rows; ++i) {
|
||||
auto index = col * i;
|
||||
if (col >= maxCol - 1) data[ 0 ].key[ index ] = { 255, 0, 0, 100 };
|
||||
else data[ 0 ].key[ index ] = { 0, 0, 255, 80 };
|
||||
}
|
||||
}
|
||||
// delete map! ;)
|
||||
lastVal = val;
|
||||
keyboard->send_led_map(data, true);
|
||||
}
|
||||
|
||||
const char *Loudness::name() {
|
||||
return m_name.c_str();
|
||||
}
|
||||
}
|
43
src/VulcanoLE/Scripts/Spectrum.cpp
Normal file
43
src/VulcanoLE/Scripts/Spectrum.cpp
Normal file
|
@ -0,0 +1,43 @@
|
|||
#include <VulcanoLE/Scripts/Spectrum.h>
|
||||
|
||||
namespace VIZ {
|
||||
Spectrum::Spectrum(AudioGrabber *pGrabber, Vulcan121 *pVulcan121) : VIZ(pGrabber, pVulcan121) {}
|
||||
|
||||
void Spectrum::on_setup() {
|
||||
keyboard->send_led_to({ 0, 0, 0, 0 });
|
||||
grabber->requestMode = AudioGrabber::ReqMode::FFT;
|
||||
usleep(100000);
|
||||
}
|
||||
void Spectrum::on_tick() {
|
||||
auto map = Vulcan121::createEmptyLEDMap();
|
||||
auto data = grabber->fft.getData()->leftChannel;
|
||||
// find largest bass ~43hz (44100 / FFT_SIZE) range
|
||||
auto val = 0.0;
|
||||
for (int i = 1; i < 4; ++i) {
|
||||
if (data[ i ] > val) {
|
||||
val = data[ i ];
|
||||
}
|
||||
}
|
||||
double newVal = (val + lastVal) / 2.0;
|
||||
lastVal = val;
|
||||
int split = keyboardData.num_keys / 3;
|
||||
int x = 0;
|
||||
int colorInd = 0;
|
||||
colors[0].a = newVal;
|
||||
colors[1].a = newVal;
|
||||
colors[2].a = newVal;
|
||||
for (int key = 0; key < keyboardData.num_keys; ++key) {
|
||||
map[ 0 ].key[ key ] = colors[ colorInd ];
|
||||
x++;
|
||||
if (x > split) {
|
||||
colorInd++;
|
||||
x = 0;
|
||||
}
|
||||
}
|
||||
keyboard->send_led_map(map, true);
|
||||
}
|
||||
|
||||
const char *Spectrum::name() {
|
||||
return m_name.c_str();
|
||||
}
|
||||
}
|
65
src/VulcanoLE/Scripts/WeirdSpec.cpp
Normal file
65
src/VulcanoLE/Scripts/WeirdSpec.cpp
Normal file
|
@ -0,0 +1,65 @@
|
|||
#include <VulcanoLE/Scripts/WeirdSpec.h>
|
||||
|
||||
namespace VIZ {
|
||||
WeirdSpec::WeirdSpec(AudioGrabber *pGrabber, Vulcan121 *pVulcan121) : VIZ(pGrabber, pVulcan121) {
|
||||
|
||||
}
|
||||
|
||||
void WeirdSpec::on_setup() {
|
||||
keyboard->send_led_to({ 0, 0, 0, 0 });
|
||||
grabber->requestMode = AudioGrabber::ReqMode::FFT;
|
||||
usleep(100000);
|
||||
}
|
||||
|
||||
void WeirdSpec::on_tick() {
|
||||
auto map = Vulcan121::createEmptyLEDMap();
|
||||
auto data = grabber->fft.getData()->leftChannel;
|
||||
auto val = 0.0;
|
||||
for (int i = 1; i < 4; ++i) {
|
||||
if (data[ i ] > val) {
|
||||
val = data[ i ];
|
||||
}
|
||||
}
|
||||
switchOnPeak(val);
|
||||
tick++;
|
||||
int colorInd = left ? 0 : 1;
|
||||
colors[ colorInd ].a = val;
|
||||
if (left) {
|
||||
for (int i = 0; i < 62; ++i) {
|
||||
int ind = i;
|
||||
map[ 0 ].key[ ind ] = colors[ colorInd ];
|
||||
}
|
||||
} else {
|
||||
for (int i = 62; i < keyboardData.num_keys; ++i) {
|
||||
int ind = i;
|
||||
map[ 0 ].key[ ind ] = colors[ colorInd ];
|
||||
}
|
||||
}
|
||||
keyboard->send_led_map(map, true);
|
||||
}
|
||||
|
||||
void WeirdSpec::switchOnPeak(double peak) {
|
||||
if (tick < 3) {
|
||||
return;
|
||||
}
|
||||
if (peak < 20) {
|
||||
lastPeak = -1;
|
||||
return;
|
||||
}
|
||||
// we dont have any data! reset ;)
|
||||
if (lastPeak == -1) {
|
||||
lastPeak = peak;
|
||||
return;
|
||||
}
|
||||
lastPeak -= decayRate;
|
||||
if (peak > 100 && peak > lastPeak + threshold) {
|
||||
left = !left;
|
||||
lastPeak = peak;
|
||||
tick = 0;
|
||||
}
|
||||
}
|
||||
|
||||
const char *WeirdSpec::name() {
|
||||
return m_name.c_str();
|
||||
}
|
||||
}
|
59
src/VulcanoLE/Visual/VisPlugins.cpp
Normal file
59
src/VulcanoLE/Visual/VisPlugins.cpp
Normal file
|
@ -0,0 +1,59 @@
|
|||
#include <VulcanoLE/Visual/VisPlugins.h>
|
||||
#include <VulcanoLE/Scripts/Spectrum.h>
|
||||
#include <VulcanoLE/Scripts/Loudness.h>
|
||||
#include <VulcanoLE/Scripts/WeirdSpec.h>
|
||||
#include <VUtils/Logging.h>
|
||||
|
||||
namespace VIZ {
|
||||
void VisPlugins::init(HIDHelper *hidHelper, AudioGrabber *audioGrabber) {
|
||||
grabber = audioGrabber;
|
||||
keyboard = new Vulcan121(hidHelper);
|
||||
viz[0] = new Spectrum(grabber, keyboard);
|
||||
viz[1] = new Loudness(grabber, keyboard);
|
||||
viz[2] = new WeirdSpec(grabber, keyboard);
|
||||
currentVis = viz[mode];
|
||||
}
|
||||
|
||||
void VisPlugins::on_startup() {
|
||||
if (!keyboard->send_init_sequence()) {
|
||||
ERR("FAILED TO INIT KEYBOARD")
|
||||
exit(1);
|
||||
}
|
||||
currentVis->on_setup();
|
||||
}
|
||||
|
||||
void VisPlugins::on_tick() {
|
||||
currentVis->on_tick();
|
||||
usleep(1000);
|
||||
}
|
||||
|
||||
void VisPlugins::on_shutdown() {
|
||||
int16_t r = env->getAsInt("shutdown_color_red", 0);
|
||||
int16_t g = env->getAsInt("shutdown_color_green", 0);
|
||||
int16_t b = env->getAsInt("shutdown_color_blue", 150);
|
||||
int16_t a = env->getAsInt("shutdown_brightness", 100);
|
||||
keyboard->send_led_to({ r, g, b, a });
|
||||
}
|
||||
|
||||
VisPlugins::~VisPlugins() {
|
||||
delete grabber;
|
||||
delete keyboard;
|
||||
for (auto &i : viz) {
|
||||
delete i;
|
||||
}
|
||||
}
|
||||
|
||||
void VisPlugins::setCurrentMode(int m) {
|
||||
if (m == 0 || m > VIZSIZE) {
|
||||
ERR("Mode Setting Failed >> Mode is not in the available range 1 - %d", VIZSIZE)
|
||||
return;
|
||||
}
|
||||
grabber->env->setNumber("visual_mode", m);
|
||||
m -= 1;
|
||||
currentVis = viz[m];
|
||||
LOG("Now Using: %s", currentVis->name());
|
||||
currentVis->on_setup();
|
||||
mode = m;
|
||||
}
|
||||
}
|
||||
|
Loading…
Reference in a new issue