mirror of
https://github.com/kavishdevar/librepods.git
synced 2026-04-29 09:33:04 +00:00
Compare commits
55 Commits
linux/rust
...
v0.2.0-alp
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
04ef891c4b | ||
|
|
555a000def | ||
|
|
45f53d5cd4 | ||
|
|
1425a7d7cf | ||
|
|
e8204a7750 | ||
|
|
5fbfda6115 | ||
|
|
a6a284c2ec | ||
|
|
5c0d9b5096 | ||
|
|
752b53aecd | ||
|
|
339c478564 | ||
|
|
c9dd79bb82 | ||
|
|
3e4d401223 | ||
|
|
5cd7db574a | ||
|
|
e10fe21ba5 | ||
|
|
b4deccff7e | ||
|
|
dcbbb2ce98 | ||
|
|
e5836c9869 | ||
|
|
287163e116 | ||
|
|
a75557d6dc | ||
|
|
0e1f784737 | ||
|
|
f3b1db2513 | ||
|
|
b5f0c32751 | ||
|
|
8088594df5 | ||
|
|
345b7b9051 | ||
|
|
fa30d3c09a | ||
|
|
826e395379 | ||
|
|
574c193a6d | ||
|
|
10bf2fe68c | ||
|
|
e192dc114b | ||
|
|
de82cdd8c2 | ||
|
|
a06c673400 | ||
|
|
a80680ff73 | ||
|
|
36c55169f1 | ||
|
|
93ac06b8e3 | ||
|
|
aa0898a65a | ||
|
|
ecfbcd1c02 | ||
|
|
5231b12c71 | ||
|
|
0123449d80 | ||
|
|
8a5d6087c9 | ||
|
|
f12fe90134 | ||
|
|
8fbdfd879c | ||
|
|
0a608afbe6 | ||
|
|
be362c5079 | ||
|
|
141f1e7604 | ||
|
|
1dbb36a2aa | ||
|
|
938f0d5448 | ||
|
|
e3eab3e31e | ||
|
|
55d1a69b21 | ||
|
|
f4fbcc9e88 | ||
|
|
944195b193 | ||
|
|
e8e1650145 | ||
|
|
4a4494121d | ||
|
|
3696a4e729 | ||
|
|
7356e57878 | ||
|
|
acf2b9bea7 |
@@ -16,5 +16,5 @@ indent_size = 4
|
|||||||
trim_trailing_whitespace = false
|
trim_trailing_whitespace = false
|
||||||
max_line_length = off
|
max_line_length = off
|
||||||
|
|
||||||
[*.{py,java,r,R,kt,xml,kts}]
|
[*.{py,java,r,R,kt,xml,kts,h,hpp,cpp,qml}]
|
||||||
indent_size = 4
|
indent_size = 4
|
||||||
|
|||||||
@@ -122,7 +122,7 @@ If primary is removed, mic will be changed and the secondary will be the new pri
|
|||||||
|
|
||||||
## Conversational Awareness
|
## Conversational Awareness
|
||||||
|
|
||||||
AirPods send conversational awareness packets when the person wearing them start speaking. The packet format is as follows:
|
AirPods send conversational awareness packets when the person wearing them starts speaking. The packet format is as follows:
|
||||||
|
|
||||||
```plaintext
|
```plaintext
|
||||||
04 00 04 00 4B 00 02 00 01 [level]
|
04 00 04 00 4B 00 02 00 01 [level]
|
||||||
@@ -307,7 +307,7 @@ All values are formatted as IEEE 754 floats in little endian order.
|
|||||||
|
|
||||||
## Configure Stem Long Press
|
## Configure Stem Long Press
|
||||||
|
|
||||||
I have noted all the packets sent to configure what the press and hold of the steam should do. The packets sent are specific to the current state. And are probably overwritten everytime the AirPods are connected to a new (apple) device that is not synced with icloud (i think)... So, for non-Apple device too, the configuration needs to be stored and overwritten everytime the AirPods are connected to the device. That is the only way to keep the configuration.
|
I have noted all the packets sent to configure what the press and hold of the steam should do. The packets sent are specific to the current state. And are probably overwritten everytime the AirPods are connected to a new (apple) device that is not synced with icloud (i think)... So, for non-Apple devices too, the configuration needs to be stored and overwritten everytime the AirPods are connected to the device. That is the only way to keep the configuration.
|
||||||
|
|
||||||
This is also the only way to control the configuration as the previous state needs to be known, and then the new state can be set.
|
This is also the only way to control the configuration as the previous state needs to be known, and then the new state can be set.
|
||||||
|
|
||||||
@@ -403,20 +403,3 @@ Once tracking is active, the AirPods stream sensor packets with the following co
|
|||||||
| orientation 3 | 47 | 2 |
|
| orientation 3 | 47 | 2 |
|
||||||
| Horizontal Acceleration | 51 | 2 |
|
| Horizontal Acceleration | 51 | 2 |
|
||||||
| Vertical Acceleration | 53 | 2 |
|
| Vertical Acceleration | 53 | 2 |
|
||||||
|
|
||||||
# LICENSE
|
|
||||||
|
|
||||||
LibrePods - AirPods liberated from Apple’s ecosystem
|
|
||||||
Copyright (C) 2025 LibrePods contributors
|
|
||||||
|
|
||||||
This program is free software: you can redistribute it and/or modify
|
|
||||||
it under the terms of the GNU Affero General Public License as published
|
|
||||||
by the Free Software Foundation, either version 3 of the License.
|
|
||||||
|
|
||||||
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 Affero General Public License for more details.
|
|
||||||
|
|
||||||
You should have received a copy of the GNU Affero General Public License
|
|
||||||
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
||||||
|
|||||||
@@ -1,128 +0,0 @@
|
|||||||
# Contributor Covenant Code of Conduct
|
|
||||||
|
|
||||||
## Our Pledge
|
|
||||||
|
|
||||||
We as members, contributors, and leaders pledge to make participation in our
|
|
||||||
community a harassment-free experience for everyone, regardless of age, body
|
|
||||||
size, visible or invisible disability, ethnicity, sex characteristics, gender
|
|
||||||
identity and expression, level of experience, education, socio-economic status,
|
|
||||||
nationality, personal appearance, race, religion, or sexual identity
|
|
||||||
and orientation.
|
|
||||||
|
|
||||||
We pledge to act and interact in ways that contribute to an open, welcoming,
|
|
||||||
diverse, inclusive, and healthy community.
|
|
||||||
|
|
||||||
## Our Standards
|
|
||||||
|
|
||||||
Examples of behavior that contributes to a positive environment for our
|
|
||||||
community include:
|
|
||||||
|
|
||||||
* Demonstrating empathy and kindness toward other people
|
|
||||||
* Being respectful of differing opinions, viewpoints, and experiences
|
|
||||||
* Giving and gracefully accepting constructive feedback
|
|
||||||
* Accepting responsibility and apologizing to those affected by our mistakes,
|
|
||||||
and learning from the experience
|
|
||||||
* Focusing on what is best not just for us as individuals, but for the
|
|
||||||
overall community
|
|
||||||
|
|
||||||
Examples of unacceptable behavior include:
|
|
||||||
|
|
||||||
* The use of sexualized language or imagery, and sexual attention or
|
|
||||||
advances of any kind
|
|
||||||
* Trolling, insulting or derogatory comments, and personal or political attacks
|
|
||||||
* Public or private harassment
|
|
||||||
* Publishing others' private information, such as a physical or email
|
|
||||||
address, without their explicit permission
|
|
||||||
* Other conduct which could reasonably be considered inappropriate in a
|
|
||||||
professional setting
|
|
||||||
|
|
||||||
## Enforcement Responsibilities
|
|
||||||
|
|
||||||
Community leaders are responsible for clarifying and enforcing our standards of
|
|
||||||
acceptable behavior and will take appropriate and fair corrective action in
|
|
||||||
response to any behavior that they deem inappropriate, threatening, offensive,
|
|
||||||
or harmful.
|
|
||||||
|
|
||||||
Community leaders have the right and responsibility to remove, edit, or reject
|
|
||||||
comments, commits, code, wiki edits, issues, and other contributions that are
|
|
||||||
not aligned to this Code of Conduct, and will communicate reasons for moderation
|
|
||||||
decisions when appropriate.
|
|
||||||
|
|
||||||
## Scope
|
|
||||||
|
|
||||||
This Code of Conduct applies within all community spaces, and also applies when
|
|
||||||
an individual is officially representing the community in public spaces.
|
|
||||||
Examples of representing our community include using an official e-mail address,
|
|
||||||
posting via an official social media account, or acting as an appointed
|
|
||||||
representative at an online or offline event.
|
|
||||||
|
|
||||||
## Enforcement
|
|
||||||
|
|
||||||
Instances of abusive, harassing, or otherwise unacceptable behavior may be
|
|
||||||
reported to the community leaders responsible for enforcement at
|
|
||||||
report@kavishdevar.me.
|
|
||||||
All complaints will be reviewed and investigated promptly and fairly.
|
|
||||||
|
|
||||||
All community leaders are obligated to respect the privacy and security of the
|
|
||||||
reporter of any incident.
|
|
||||||
|
|
||||||
## Enforcement Guidelines
|
|
||||||
|
|
||||||
Community leaders will follow these Community Impact Guidelines in determining
|
|
||||||
the consequences for any action they deem in violation of this Code of Conduct:
|
|
||||||
|
|
||||||
### 1. Correction
|
|
||||||
|
|
||||||
**Community Impact**: Use of inappropriate language or other behavior deemed
|
|
||||||
unprofessional or unwelcome in the community.
|
|
||||||
|
|
||||||
**Consequence**: A private, written warning from community leaders, providing
|
|
||||||
clarity around the nature of the violation and an explanation of why the
|
|
||||||
behavior was inappropriate. A public apology may be requested.
|
|
||||||
|
|
||||||
### 2. Warning
|
|
||||||
|
|
||||||
**Community Impact**: A violation through a single incident or series
|
|
||||||
of actions.
|
|
||||||
|
|
||||||
**Consequence**: A warning with consequences for continued behavior. No
|
|
||||||
interaction with the people involved, including unsolicited interaction with
|
|
||||||
those enforcing the Code of Conduct, for a specified period of time. This
|
|
||||||
includes avoiding interactions in community spaces as well as external channels
|
|
||||||
like social media. Violating these terms may lead to a temporary or
|
|
||||||
permanent ban.
|
|
||||||
|
|
||||||
### 3. Temporary Ban
|
|
||||||
|
|
||||||
**Community Impact**: A serious violation of community standards, including
|
|
||||||
sustained inappropriate behavior.
|
|
||||||
|
|
||||||
**Consequence**: A temporary ban from any sort of interaction or public
|
|
||||||
communication with the community for a specified period of time. No public or
|
|
||||||
private interaction with the people involved, including unsolicited interaction
|
|
||||||
with those enforcing the Code of Conduct, is allowed during this period.
|
|
||||||
Violating these terms may lead to a permanent ban.
|
|
||||||
|
|
||||||
### 4. Permanent Ban
|
|
||||||
|
|
||||||
**Community Impact**: Demonstrating a pattern of violation of community
|
|
||||||
standards, including sustained inappropriate behavior, harassment of an
|
|
||||||
individual, or aggression toward or disparagement of classes of individuals.
|
|
||||||
|
|
||||||
**Consequence**: A permanent ban from any sort of public interaction within
|
|
||||||
the community.
|
|
||||||
|
|
||||||
## Attribution
|
|
||||||
|
|
||||||
This Code of Conduct is adapted from the [Contributor Covenant][homepage],
|
|
||||||
version 2.0, available at
|
|
||||||
https://www.contributor-covenant.org/version/2/0/code_of_conduct.html.
|
|
||||||
|
|
||||||
Community Impact Guidelines were inspired by [Mozilla's code of conduct
|
|
||||||
enforcement ladder](https://github.com/mozilla/diversity).
|
|
||||||
|
|
||||||
[homepage]: https://www.contributor-covenant.org
|
|
||||||
|
|
||||||
For answers to common questions about this code of conduct, see the FAQ at
|
|
||||||
https://www.contributor-covenant.org/faq. Translations are available at
|
|
||||||
https://www.contributor-covenant.org/translations.
|
|
||||||
@@ -1,70 +0,0 @@
|
|||||||
# Welcome to LibrePods contributing guide <!-- omit in toc -->
|
|
||||||
|
|
||||||
Thank you for considering a contribution to LibrePods! Your support helps bring Apple-exclusive AirPods features to Linux and Android.
|
|
||||||
|
|
||||||
Read our [Code of Conduct](./CODE_OF_CONDUCT.md) to keep our community approachable and respectful.
|
|
||||||
|
|
||||||
This guide provides an overview of the contribution workflow, from opening an issue to creating and reviewing a pull request (PR).
|
|
||||||
|
|
||||||
## New contributor guide
|
|
||||||
|
|
||||||
To get an overview of the project, read the [README](./README.md). Here are some resources to help you get started with open-source contributions:
|
|
||||||
|
|
||||||
- [Finding ways to contribute to open source on GitHub](https://docs.github.com/en/get-started/exploring-projects-on-github/finding-ways-to-contribute-to-open-source-on-github)
|
|
||||||
- [Set up Git](https://docs.github.com/en/get-started/getting-started-with-git/set-up-git)
|
|
||||||
- [GitHub flow](https://docs.github.com/en/get-started/using-github/github-flow)
|
|
||||||
- [Collaborating with pull requests](https://docs.github.com/en/github/collaborating-with-pull-requests)
|
|
||||||
|
|
||||||
## Getting started
|
|
||||||
|
|
||||||
To navigate our codebase with confidence, see the [README](./README.md) for setup instructions and usage details. We accept various types of contributions, which don’t always require writing code (like translations).
|
|
||||||
|
|
||||||
To develop for the Android App, Android Studio is the preferred IDE. And you can use any IDE for the linux program, it is just python!
|
|
||||||
|
|
||||||
### Issues
|
|
||||||
|
|
||||||
#### Create a new issue
|
|
||||||
|
|
||||||
If you find a bug or want to suggest a feature, check if an issue already exists by searching through our [existing issues](https://github.com/kavishdevar/librepods/issues). If no relevant issue exists, open a new one and fill in the details.
|
|
||||||
|
|
||||||
#### Solve an issue
|
|
||||||
|
|
||||||
Browse our [issues list](https://github.com/kavishdevar/librepods/issues) to find an interesting issue to work on. Use labels to filter issues and pick one that matches your expertise. If you’d like to work on an issue, open a PR with your solution.
|
|
||||||
|
|
||||||
### Make Changes
|
|
||||||
|
|
||||||
#### Make changes locally
|
|
||||||
|
|
||||||
1. Fork the repository and clone it to your local environment.
|
|
||||||
```
|
|
||||||
git clone https://github.com/kavishdevar/librepods.git
|
|
||||||
cd AirPods-Like-Normal
|
|
||||||
```
|
|
||||||
2. Create a working branch to start your changes.
|
|
||||||
```
|
|
||||||
git checkout -b your-feature-branch
|
|
||||||
```
|
|
||||||
3. Make your changes, following the existing style and structure.
|
|
||||||
|
|
||||||
4. Test your changes to ensure they work as expected and do not introduce new issues.
|
|
||||||
|
|
||||||
### Commit your changes
|
|
||||||
|
|
||||||
Commit your changes with a descriptive message.
|
|
||||||
|
|
||||||
### Pull Request
|
|
||||||
|
|
||||||
When your changes are ready, create a pull request (PR):
|
|
||||||
- Fill out the PR template to help reviewers understand your changes.
|
|
||||||
- If your PR is related to an issue, don’t forget to [link your PR to it](https://docs.github.com/en/issues/tracking-your-work-with-issues/linking-a-pull-request-to-an-issue).
|
|
||||||
- Enable the checkbox to allow maintainers to edit your PR, so any required changes can be merged easily.
|
|
||||||
|
|
||||||
Once your PR is open, a team member will review it. They may ask questions or request additional information.
|
|
||||||
|
|
||||||
- If changes are requested, apply them in your fork and commit them to the PR branch.
|
|
||||||
- Mark conversations as resolved as you apply feedback.
|
|
||||||
- For merge conflicts, follow this [git tutorial](https://github.com/skills/resolve-merge-conflicts) to resolve them.
|
|
||||||
|
|
||||||
### Your PR is merged!
|
|
||||||
|
|
||||||
Congratulations! :tada: Once merged, your contributions will be publicly available in LibrePods.
|
|
||||||
147
LICENSE
147
LICENSE
@@ -1,5 +1,5 @@
|
|||||||
GNU AFFERO GENERAL PUBLIC LICENSE
|
GNU GENERAL PUBLIC LICENSE
|
||||||
Version 3, 19 November 2007
|
Version 3, 29 June 2007
|
||||||
|
|
||||||
Copyright (C) 2007 Free Software Foundation, Inc. <https://fsf.org/>
|
Copyright (C) 2007 Free Software Foundation, Inc. <https://fsf.org/>
|
||||||
Everyone is permitted to copy and distribute verbatim copies
|
Everyone is permitted to copy and distribute verbatim copies
|
||||||
@@ -7,15 +7,17 @@
|
|||||||
|
|
||||||
Preamble
|
Preamble
|
||||||
|
|
||||||
The GNU Affero General Public License is a free, copyleft license for
|
The GNU General Public License is a free, copyleft license for
|
||||||
software and other kinds of works, specifically designed to ensure
|
software and other kinds of works.
|
||||||
cooperation with the community in the case of network server software.
|
|
||||||
|
|
||||||
The licenses for most software and other practical works are designed
|
The licenses for most software and other practical works are designed
|
||||||
to take away your freedom to share and change the works. By contrast,
|
to take away your freedom to share and change the works. By contrast,
|
||||||
our General Public Licenses are intended to guarantee your freedom to
|
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
|
share and change all versions of a program--to make sure it remains free
|
||||||
software for all its users.
|
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
|
When we speak of free software, we are referring to freedom, not
|
||||||
price. Our General Public Licenses are designed to make sure that you
|
price. Our General Public Licenses are designed to make sure that you
|
||||||
@@ -24,34 +26,44 @@ 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
|
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.
|
free programs, and that you know you can do these things.
|
||||||
|
|
||||||
Developers that use our General Public Licenses protect your rights
|
To protect your rights, we need to prevent others from denying you
|
||||||
with two steps: (1) assert copyright on the software, and (2) offer
|
these rights or asking you to surrender the rights. Therefore, you have
|
||||||
you this License which gives you legal permission to copy, distribute
|
certain responsibilities if you distribute copies of the software, or if
|
||||||
and/or modify the software.
|
you modify it: responsibilities to respect the freedom of others.
|
||||||
|
|
||||||
A secondary benefit of defending all users' freedom is that
|
For example, if you distribute copies of such a program, whether
|
||||||
improvements made in alternate versions of the program, if they
|
gratis or for a fee, you must pass on to the recipients the same
|
||||||
receive widespread use, become available for other developers to
|
freedoms that you received. You must make sure that they, too, receive
|
||||||
incorporate. Many developers of free software are heartened and
|
or can get the source code. And you must show them these terms so they
|
||||||
encouraged by the resulting cooperation. However, in the case of
|
know their rights.
|
||||||
software used on network servers, this result may fail to come about.
|
|
||||||
The GNU General Public License permits making a modified version and
|
|
||||||
letting the public access it on a server without ever releasing its
|
|
||||||
source code to the public.
|
|
||||||
|
|
||||||
The GNU Affero General Public License is designed specifically to
|
Developers that use the GNU GPL protect your rights with two steps:
|
||||||
ensure that, in such cases, the modified source code becomes available
|
(1) assert copyright on the software, and (2) offer you this License
|
||||||
to the community. It requires the operator of a network server to
|
giving you legal permission to copy, distribute and/or modify it.
|
||||||
provide the source code of the modified version running there to the
|
|
||||||
users of that server. Therefore, public use of a modified version, on
|
|
||||||
a publicly accessible server, gives the public access to the source
|
|
||||||
code of the modified version.
|
|
||||||
|
|
||||||
An older license, called the Affero General Public License and
|
For the developers' and authors' protection, the GPL clearly explains
|
||||||
published by Affero, was designed to accomplish similar goals. This is
|
that there is no warranty for this free software. For both users' and
|
||||||
a different license, not a version of the Affero GPL, but Affero has
|
authors' sake, the GPL requires that modified versions be marked as
|
||||||
released a new version of the Affero GPL which permits relicensing under
|
changed, so that their problems will not be attributed erroneously to
|
||||||
this license.
|
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
|
The precise terms and conditions for copying, distribution and
|
||||||
modification follow.
|
modification follow.
|
||||||
@@ -60,7 +72,7 @@ modification follow.
|
|||||||
|
|
||||||
0. Definitions.
|
0. Definitions.
|
||||||
|
|
||||||
"This License" refers to version 3 of the GNU Affero General Public License.
|
"This License" refers to version 3 of the GNU General Public License.
|
||||||
|
|
||||||
"Copyright" also means copyright-like laws that apply to other kinds of
|
"Copyright" also means copyright-like laws that apply to other kinds of
|
||||||
works, such as semiconductor masks.
|
works, such as semiconductor masks.
|
||||||
@@ -537,45 +549,35 @@ 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
|
the Program, the only way you could satisfy both those terms and this
|
||||||
License would be to refrain entirely from conveying the Program.
|
License would be to refrain entirely from conveying the Program.
|
||||||
|
|
||||||
13. Remote Network Interaction; Use with the GNU General Public License.
|
13. Use with the GNU Affero General Public License.
|
||||||
|
|
||||||
Notwithstanding any other provision of this License, if you modify the
|
|
||||||
Program, your modified version must prominently offer all users
|
|
||||||
interacting with it remotely through a computer network (if your version
|
|
||||||
supports such interaction) an opportunity to receive the Corresponding
|
|
||||||
Source of your version by providing access to the Corresponding Source
|
|
||||||
from a network server at no charge, through some standard or customary
|
|
||||||
means of facilitating copying of software. This Corresponding Source
|
|
||||||
shall include the Corresponding Source for any work covered by version 3
|
|
||||||
of the GNU General Public License that is incorporated pursuant to the
|
|
||||||
following paragraph.
|
|
||||||
|
|
||||||
Notwithstanding any other provision of this License, you have
|
Notwithstanding any other provision of this License, you have
|
||||||
permission to link or combine any covered work with a work licensed
|
permission to link or combine any covered work with a work licensed
|
||||||
under version 3 of the GNU General Public License into a single
|
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
|
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,
|
License will continue to apply to the part which is the covered work,
|
||||||
but the work with which it is combined will remain governed by version
|
but the special requirements of the GNU Affero General Public License,
|
||||||
3 of the GNU General Public License.
|
section 13, concerning interaction through a network will apply to the
|
||||||
|
combination as such.
|
||||||
|
|
||||||
14. Revised Versions of this License.
|
14. Revised Versions of this License.
|
||||||
|
|
||||||
The Free Software Foundation may publish revised and/or new versions of
|
The Free Software Foundation may publish revised and/or new versions of
|
||||||
the GNU Affero General Public License from time to time. Such new versions
|
the GNU General Public License from time to time. Such new versions will
|
||||||
will be similar in spirit to the present version, but may differ in detail to
|
be similar in spirit to the present version, but may differ in detail to
|
||||||
address new problems or concerns.
|
address new problems or concerns.
|
||||||
|
|
||||||
Each version is given a distinguishing version number. If the
|
Each version is given a distinguishing version number. If the
|
||||||
Program specifies that a certain numbered version of the GNU Affero General
|
Program specifies that a certain numbered version of the GNU General
|
||||||
Public License "or any later version" applies to it, you have the
|
Public License "or any later version" applies to it, you have the
|
||||||
option of following the terms and conditions either of that numbered
|
option of following the terms and conditions either of that numbered
|
||||||
version or of any later version published by the Free Software
|
version or of any later version published by the Free Software
|
||||||
Foundation. If the Program does not specify a version number of the
|
Foundation. If the Program does not specify a version number of the
|
||||||
GNU Affero General Public License, you may choose any version ever published
|
GNU General Public License, you may choose any version ever published
|
||||||
by the Free Software Foundation.
|
by the Free Software Foundation.
|
||||||
|
|
||||||
If the Program specifies that a proxy can decide which future
|
If the Program specifies that a proxy can decide which future
|
||||||
versions of the GNU Affero General Public License can be used, that proxy's
|
versions of the GNU General Public License can be used, that proxy's
|
||||||
public statement of acceptance of a version permanently authorizes you
|
public statement of acceptance of a version permanently authorizes you
|
||||||
to choose that version for the Program.
|
to choose that version for the Program.
|
||||||
|
|
||||||
@@ -633,29 +635,40 @@ the "copyright" line and a pointer to where the full notice is found.
|
|||||||
Copyright (C) <year> <name of author>
|
Copyright (C) <year> <name of author>
|
||||||
|
|
||||||
This program is free software: you can redistribute it and/or modify
|
This program is free software: you can redistribute it and/or modify
|
||||||
it under the terms of the GNU Affero General Public License as published
|
it under the terms of the GNU General Public License as published by
|
||||||
by the Free Software Foundation, either version 3 of the License, or
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
(at your option) any later version.
|
any later version.
|
||||||
|
|
||||||
This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
|
|
||||||
You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
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.
|
Also add information on how to contact you by electronic and paper mail.
|
||||||
|
|
||||||
If your software can interact with users remotely through a computer
|
If the program does terminal interaction, make it output a short
|
||||||
network, you should also make sure that it provides a way for users to
|
notice like this when it starts in an interactive mode:
|
||||||
get its source. For example, if your program is a web application, its
|
|
||||||
interface could display a "Source" link that leads users to an archive
|
<program> Copyright (C) <year> <name of author>
|
||||||
of the code. There are many ways you could offer source, and different
|
This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
|
||||||
solutions will be better for different programs; see section 13 for the
|
This is free software, and you are welcome to redistribute it
|
||||||
specific requirements.
|
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,
|
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.
|
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 AGPL, see
|
For more information on this, and how to apply and follow the GNU GPL, see
|
||||||
<https://www.gnu.org/licenses/>.
|
<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>.
|
||||||
101
README.md
101
README.md
@@ -1,13 +1,7 @@
|
|||||||

|
>[!IMPORTANT]
|
||||||
|
Development paused due to lack of time until 17th May 2026 (JEE Advanced). PRs and issues might not be responded to until then.
|
||||||
[](https://xdaforums.com/t/app-root-for-now-airpodslikenormal-unlock-apple-exclusive-airpods-features-on-android.4707585/)
|
|
||||||
[](https://github.com/kavishdevar/librepods/releases/latest)
|
|
||||||
[](https://github.com/kavishdevar/librepods/releases)
|
|
||||||
[](https://github.com/kavishdevar/librepods/stargazers)
|
|
||||||
[](https://github.com/kavishdevar/librepods/issues)
|
|
||||||
[](https://github.com/kavishdevar/librepods/blob/main/LICENSE)
|
|
||||||
[](https://github.com/kavishdevar/librepods/graphs/contributors)
|
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
## What is LibrePods?
|
## What is LibrePods?
|
||||||
|
|
||||||
@@ -19,9 +13,10 @@ LibrePods unlocks Apple's exclusive AirPods features on non-Apple devices. Get a
|
|||||||
| ------ | --------------------- | ---------------------------------------------------------- |
|
| ------ | --------------------- | ---------------------------------------------------------- |
|
||||||
| ✅ | AirPods Pro (2nd Gen) | Fully supported and tested |
|
| ✅ | AirPods Pro (2nd Gen) | Fully supported and tested |
|
||||||
| ✅ | AirPods Pro (3rd Gen) | Fully supported (except heartrate monitoring) |
|
| ✅ | AirPods Pro (3rd Gen) | Fully supported (except heartrate monitoring) |
|
||||||
|
| ✅ | AirPods Max | Fully supported (client shows unsupported features) |
|
||||||
| ⚠️ | Other AirPods models | Basic features (battery status, ear detection) should work |
|
| ⚠️ | Other AirPods models | Basic features (battery status, ear detection) should work |
|
||||||
|
|
||||||
Most features should work with any AirPods. Currently, I've only got AirPods Pro 2 to test with.
|
Most features should work with any AirPods. Currently, I've only got AirPods Pro 2 to test with. But, I believe the protocol remains the same for all other AirPods (based on analysis of the bluetooth stack on macOS).
|
||||||
|
|
||||||
## Key Features
|
## Key Features
|
||||||
|
|
||||||
@@ -36,39 +31,31 @@ Most features should work with any AirPods. Currently, I've only got AirPods Pro
|
|||||||
- **Other customizations**:
|
- **Other customizations**:
|
||||||
- Rename your AirPods
|
- Rename your AirPods
|
||||||
- Customize long-press actions
|
- Customize long-press actions
|
||||||
- Few accessibility features
|
- All accessibility settings
|
||||||
- And more!
|
- And more!
|
||||||
|
|
||||||
See our [pinned issue](https://github.com/kavishdevar/librepods/issues/20) for a complete feature list and roadmap.
|
* Features marked with an asterisk require the VendorID to be change to that of Apple.
|
||||||
|
|
||||||
## Platform Support
|
## Platform Support
|
||||||
|
|
||||||
### Linux
|
### Linux
|
||||||
|
for the old version see the [Linux README](./linux/README.md). (doesn't have many features, maintainer didn't have time to work on it)
|
||||||
|
|
||||||
The Linux version runs as a system tray app. Connect your AirPods and enjoy:
|
new version in development ([#241](https://github.com/kavishdevar/librepods/pull/241))
|
||||||
|
|
||||||
- Battery monitoring
|

|
||||||
- Automatic Ear detection
|
|
||||||
- Conversational Awareness
|
|
||||||
- Switching Noise Control modes
|
|
||||||
- Device renaming
|
|
||||||
|
|
||||||
> [!NOTE]
|
|
||||||
> Work in progress, but core functionality is stable and usable.
|
|
||||||
|
|
||||||
For installation and detailed info, see the [Linux README](/linux/README.md).
|
|
||||||
|
|
||||||
### Android
|
### Android
|
||||||
|
|
||||||
#### Screenshots
|
#### Screenshots
|
||||||
|
|
||||||
| | | |
|
| | | |
|
||||||
| -------------------------------------------------------------------------------------- | ------------------------------------------------- | --------------------------------------------------------------------------- |
|
| --------------------------------------------------------------------------------------- | -------------------------------------------------- | ---------------------------------------------------------------------------- |
|
||||||
|  |  |  |
|
|  |  |  |
|
||||||
|  |  |  |
|
|  |  |  |
|
||||||
|  |  |  |
|
|  |  |  |
|
||||||
|  |  |  |
|
|  |  |  |
|
||||||
|  |  |  |
|
|  |  |  |
|
||||||
|
|
||||||
|
|
||||||
here's a very unprofessional demo video
|
here's a very unprofessional demo video
|
||||||
@@ -77,16 +64,20 @@ https://github.com/user-attachments/assets/43911243-0576-4093-8c55-89c1db5ea533
|
|||||||
|
|
||||||
#### Root Requirement
|
#### Root Requirement
|
||||||
|
|
||||||
|
If you are using ColorOS/OxygenOS 16, you don't need root except for customizing transparency mode, setting up hearing aid, and use Bluetooth Multipoint. Changing ANC, conversational awareness, ear detection, and other customizations will work without root. For everyone else:
|
||||||
|
|
||||||
> [!CAUTION]
|
> [!CAUTION]
|
||||||
> **You must have a rooted device with Xposed to use LibrePods on Android.** This is due to a [bug in the Android Bluetooth stack](https://issuetracker.google.com/issues/371713238). Please upvote the issue by clicking the '+1' icon on the IssueTracker page.
|
> **You must have a rooted device with Xposed to use LibrePods on Android.** This is due to a [bug in the Android Bluetooth stack](https://issuetracker.google.com/issues/371713238). Please upvote the issue by clicking the '+1' icon on the IssueTracker page. DO NOT leave a +1 comment - use the +1 button in the top right of the page next to the "Hotlists" field. Leaving +1 comment spam makes it impossible for developers to engage in the necessary technical discussion to implement this fix, and will disincentivize the responsible Google developers from engaging. I don't know a fix for Android versions <13 either. So, this needs a phone running A13+.
|
||||||
>
|
>
|
||||||
> There are **no exceptions** to the root requirement until Google merges the fix.
|
> There are **no exceptions** to the root requirement until Google/your OEM figures out a fix.
|
||||||
|
|
||||||
Until then, you must xposed. I used to provide a non-xposed method too, where the module used overlayfs to replace the bluetooth library with a locally patched one, but that was broken due to how various devices handled overlayfs and a patched library. With xposed, you can also enable the DID hook enabling a few extra features.
|
Until then, you must xposed. I used to provide a non-xposed method too, where the module used overlayfs to replace the bluetooth library with a locally patched one, but that was broken due to how various devices handled overlayfs and a patched library. With xposed, you can also enable the DID hook enabling a few extra features.
|
||||||
|
|
||||||
## Bluetooth DID (Device Identification) Hook
|
## Changing VendorID in the DID profile to that of Apple
|
||||||
|
|
||||||
Turns out, if you change the manufacturerid to that of Apple, you get access to several special features!
|
Turns out, if you change the VendorID in DID Profile to that of Apple, you get access to several special features!
|
||||||
|
|
||||||
|
You can do this on Linux by editing the DeviceID in `/etc/bluetooth/main.conf`. Add this line to the config file `DeviceID = bluetooth:004C:0000:0000`. For android you can enable the `act as Apple device` setting in the app's settings.
|
||||||
|
|
||||||
### Multi-device Connectivity
|
### Multi-device Connectivity
|
||||||
|
|
||||||
@@ -96,13 +87,11 @@ Upto two devices can be simultaneously connected to AirPods, for audio and contr
|
|||||||
|
|
||||||
Accessibility settings like customizing transparency mode (amplification, balance, tone, conversation boost, and ambient noise reduction), and loud sound reduction can be configured.
|
Accessibility settings like customizing transparency mode (amplification, balance, tone, conversation boost, and ambient noise reduction), and loud sound reduction can be configured.
|
||||||
|
|
||||||
All hearing aid customizations can be done from Android, including setting the audiogram result. The app doesn't provide a way to take a hearing test because it requires much more precision. It is much better to use an already available audiogram result.
|
All hearing aid customizations can be done from Android (linux soon), including setting the audiogram result. The app doesn't provide a way to take a hearing test because it requires much more precision. It is much better to use an already available audiogram result.
|
||||||
|
|
||||||
To enable these features, enable App Settings -> `act as Apple Device`.
|
|
||||||
|
|
||||||
#### A few notes
|
#### A few notes
|
||||||
|
|
||||||
- Due to recent AirPods' firmware upgrades, you must enable `Off listening mode` to switch to `Off`. This is because in this mode, louds sounds are not reduced.
|
- Due to recent AirPods' firmware upgrades, you must enable `Off listening mode` to switch to `Off`. This is because in this mode, loud sounds are not reduced.
|
||||||
|
|
||||||
- If you have take both AirPods out, the app will automatically switch to the phone speaker. But, Android might keep on trying to connect to the AirPods because the phone is still connected to them, just the A2DP profile is not connected. The app tries to disconnect the A2DP profile as soon as it detects that Android has connected again if they're not in the ear.
|
- If you have take both AirPods out, the app will automatically switch to the phone speaker. But, Android might keep on trying to connect to the AirPods because the phone is still connected to them, just the A2DP profile is not connected. The app tries to disconnect the A2DP profile as soon as it detects that Android has connected again if they're not in the ear.
|
||||||
|
|
||||||
@@ -110,9 +99,32 @@ To enable these features, enable App Settings -> `act as Apple Device`.
|
|||||||
|
|
||||||
- If you want the AirPods icon and battery status to show in Android Settings app, install the app as a system app by using the root module.
|
- If you want the AirPods icon and battery status to show in Android Settings app, install the app as a system app by using the root module.
|
||||||
|
|
||||||
|
## Supporters
|
||||||
|
|
||||||
|
A huge thank you to everyone supporting the project!
|
||||||
|
- @davdroman
|
||||||
|
- @tedsalmon
|
||||||
|
- @wiless
|
||||||
|
- @SmartMsg
|
||||||
|
- @lunaroyster
|
||||||
|
- @ressiwage
|
||||||
|
|
||||||
|
## Special thanks
|
||||||
|
- @tyalie for making the first documentation on the protocol! ([tyalie/AAP-Protocol-Definition](https://github.com/tyalie/AAP-Protocol-Defintion))
|
||||||
|
- @rithvikvibhu and folks over at lagrangepoint for helping with the hearing aid feature ([gist](https://gist.github.com/rithvikvibhu/45e24bbe5ade30125f152383daf07016))
|
||||||
|
- @devnoname120 for helping with the first root patch
|
||||||
|
- @timgromeyer for making the first version of the linux app
|
||||||
|
- @hackclub for hosting [High Seas](https://highseas.hackclub.com) and [Low Skies](https://low-skies.hackclub.com)!
|
||||||
|
|
||||||
## Star History
|
## Star History
|
||||||
|
|
||||||
[](https://star-history.com/#kavishdevar/librepods&Date)
|
<a href="https://www.star-history.com/#kavishdevar/librepods&type=date&legend=top-left">
|
||||||
|
<picture>
|
||||||
|
<source media="(prefers-color-scheme: dark)" srcset="https://api.star-history.com/svg?repos=kavishdevar/librepods&type=date&theme=dark&legend=top-left" />
|
||||||
|
<source media="(prefers-color-scheme: light)" srcset="https://api.star-history.com/svg?repos=kavishdevar/librepods&type=date&legend=top-left" />
|
||||||
|
<img alt="Star History Chart" src="https://api.star-history.com/svg?repos=kavishdevar/librepods&type=date&legend=top-left" />
|
||||||
|
</picture>
|
||||||
|
</a>
|
||||||
|
|
||||||
# License
|
# License
|
||||||
|
|
||||||
@@ -120,15 +132,16 @@ LibrePods - AirPods liberated from Apple’s ecosystem
|
|||||||
Copyright (C) 2025 LibrePods contributors
|
Copyright (C) 2025 LibrePods contributors
|
||||||
|
|
||||||
This program is free software: you can redistribute it and/or modify
|
This program is free software: you can redistribute it and/or modify
|
||||||
it under the terms of the GNU Affero General Public License as published
|
it under the terms of the GNU General Public License as published by
|
||||||
by the Free Software Foundation, either version 3 of the License.
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
|
any later version.
|
||||||
|
|
||||||
This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
|
|
||||||
You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
along with this program over [here](/LICENSE). If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
All trademarks, logos, and brand names are the property of their respective owners. Use of them does not imply any affiliation with or endorsement by them. All AirPods images, symbols, and the SF Pro font are the property of Apple Inc.
|
All trademarks, logos, and brand names are the property of their respective owners. Use of them does not imply any affiliation with or endorsement by them. All AirPods images, symbols, and the SF Pro font are the property of Apple Inc.
|
||||||
|
|||||||
@@ -12,10 +12,10 @@ android {
|
|||||||
|
|
||||||
defaultConfig {
|
defaultConfig {
|
||||||
applicationId = "me.kavishdevar.librepods"
|
applicationId = "me.kavishdevar.librepods"
|
||||||
minSdk = 28
|
minSdk = 33
|
||||||
targetSdk = 36
|
targetSdk = 36
|
||||||
versionCode = 8
|
versionCode = 10
|
||||||
versionName = "0.2.0"
|
versionName = "0.2.0-alpha.2"
|
||||||
}
|
}
|
||||||
|
|
||||||
buildTypes {
|
buildTypes {
|
||||||
@@ -38,6 +38,9 @@ android {
|
|||||||
compose = true
|
compose = true
|
||||||
viewBinding = true
|
viewBinding = true
|
||||||
}
|
}
|
||||||
|
androidResources {
|
||||||
|
generateLocaleConfig = true
|
||||||
|
}
|
||||||
externalNativeBuild {
|
externalNativeBuild {
|
||||||
cmake {
|
cmake {
|
||||||
path = file("src/main/cpp/CMakeLists.txt")
|
path = file("src/main/cpp/CMakeLists.txt")
|
||||||
|
|||||||
@@ -10,8 +10,6 @@
|
|||||||
<uses-permission android:name="android.permission.FOREGROUND_SERVICE"
|
<uses-permission android:name="android.permission.FOREGROUND_SERVICE"
|
||||||
tools:ignore="ForegroundServicesPolicy" />
|
tools:ignore="ForegroundServicesPolicy" />
|
||||||
<uses-permission android:name="android.permission.FOREGROUND_SERVICE_CONNECTED_DEVICE" />
|
<uses-permission android:name="android.permission.FOREGROUND_SERVICE_CONNECTED_DEVICE" />
|
||||||
<uses-permission android:name="android.permission.INTERACT_ACROSS_USERS"
|
|
||||||
tools:ignore="ProtectedPermissions" />
|
|
||||||
<uses-permission android:name="android.permission.BLUETOOTH_ADMIN" />
|
<uses-permission android:name="android.permission.BLUETOOTH_ADMIN" />
|
||||||
<uses-permission android:name="android.permission.POST_NOTIFICATIONS" />
|
<uses-permission android:name="android.permission.POST_NOTIFICATIONS" />
|
||||||
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
|
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
|
||||||
|
|||||||
@@ -3,10 +3,32 @@ cmake_minimum_required(VERSION 3.22.1)
|
|||||||
project("l2c_fcr_hook")
|
project("l2c_fcr_hook")
|
||||||
set(CMAKE_CXX_STANDARD 23)
|
set(CMAKE_CXX_STANDARD 23)
|
||||||
|
|
||||||
add_library(${CMAKE_PROJECT_NAME} SHARED
|
add_library(l2c_fcr_hook SHARED
|
||||||
l2c_fcr_hook.cpp
|
l2c_fcr_hook.cpp
|
||||||
l2c_fcr_hook.h)
|
|
||||||
|
|
||||||
target_link_libraries(${CMAKE_PROJECT_NAME}
|
xz/xz_crc32.c
|
||||||
|
xz/xz_crc64.c
|
||||||
|
xz/xz_sha256.c
|
||||||
|
xz/xz_dec_stream.c
|
||||||
|
xz/xz_dec_lzma2.c
|
||||||
|
xz/xz_dec_bcj.c
|
||||||
|
)
|
||||||
|
|
||||||
|
target_include_directories(l2c_fcr_hook PRIVATE
|
||||||
|
xz
|
||||||
|
)
|
||||||
|
|
||||||
|
target_compile_definitions(l2c_fcr_hook PRIVATE
|
||||||
|
XZ_DEC_X86
|
||||||
|
XZ_DEC_ARM
|
||||||
|
XZ_DEC_ARMTHUMB
|
||||||
|
XZ_DEC_ARM64
|
||||||
|
XZ_DEC_ANY_CHECK
|
||||||
|
XZ_USE_CRC64
|
||||||
|
XZ_USE_SHA256
|
||||||
|
XZ_DEC_CONCATENATED
|
||||||
|
)
|
||||||
|
|
||||||
|
target_link_libraries(l2c_fcr_hook
|
||||||
android
|
android
|
||||||
log)
|
log)
|
||||||
|
|||||||
@@ -1,491 +1,324 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <cstdint>
|
|
||||||
#include <cstring>
|
|
||||||
#include <dlfcn.h>
|
|
||||||
#include <android/log.h>
|
#include <android/log.h>
|
||||||
#include <fstream>
|
#include <cstring>
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <sys/system_properties.h>
|
#include <vector>
|
||||||
#include "l2c_fcr_hook.h"
|
#include <fcntl.h>
|
||||||
#include <cerrno>
|
#include <unistd.h>
|
||||||
#include <cstdlib>
|
#include <sys/stat.h>
|
||||||
|
#include <elf.h>
|
||||||
|
|
||||||
#define LOG_TAG "AirPodsHook"
|
#include "l2c_fcr_hook.h"
|
||||||
|
|
||||||
|
extern "C" {
|
||||||
|
#include "xz.h"
|
||||||
|
}
|
||||||
|
|
||||||
|
#define LOG_TAG "LibrePods"
|
||||||
#define LOGI(...) __android_log_print(ANDROID_LOG_INFO, LOG_TAG, __VA_ARGS__)
|
#define LOGI(...) __android_log_print(ANDROID_LOG_INFO, LOG_TAG, __VA_ARGS__)
|
||||||
#define LOGE(...) __android_log_print(ANDROID_LOG_ERROR, LOG_TAG, __VA_ARGS__)
|
#define LOGE(...) __android_log_print(ANDROID_LOG_ERROR, LOG_TAG, __VA_ARGS__)
|
||||||
|
|
||||||
static HookFunType hook_func = nullptr;
|
static HookFunType hook_func = nullptr;
|
||||||
#define L2CEVT_L2CAP_CONFIG_REQ 4
|
|
||||||
#define L2CEVT_L2CAP_CONFIG_RSP 15
|
|
||||||
|
|
||||||
struct t_l2c_lcb;
|
static uint8_t (*original_l2c_fcr_chk_chan_modes)(void*) = nullptr;
|
||||||
typedef struct _BT_HDR {
|
static tBTA_STATUS (*original_BTA_DmSetLocalDiRecord)(
|
||||||
uint16_t event;
|
tSDP_DI_RECORD*, uint32_t*) = nullptr;
|
||||||
uint16_t len;
|
|
||||||
uint16_t offset;
|
|
||||||
uint16_t layer_specific;
|
|
||||||
uint8_t data[];
|
|
||||||
} BT_HDR;
|
|
||||||
|
|
||||||
typedef struct {
|
|
||||||
uint8_t mode;
|
|
||||||
uint8_t tx_win_sz;
|
|
||||||
uint8_t max_transmit;
|
|
||||||
uint16_t rtrans_tout;
|
|
||||||
uint16_t mon_tout;
|
|
||||||
uint16_t mps;
|
|
||||||
} tL2CAP_FCR;
|
|
||||||
|
|
||||||
// Flow spec structure
|
|
||||||
typedef struct {
|
|
||||||
uint8_t qos_present;
|
|
||||||
uint8_t flow_direction;
|
|
||||||
uint8_t service_type;
|
|
||||||
uint32_t token_rate;
|
|
||||||
uint32_t token_bucket_size;
|
|
||||||
uint32_t peak_bandwidth;
|
|
||||||
uint32_t latency;
|
|
||||||
uint32_t delay_variation;
|
|
||||||
} FLOW_SPEC;
|
|
||||||
|
|
||||||
// Configuration info structure
|
|
||||||
typedef struct {
|
|
||||||
uint16_t result;
|
|
||||||
uint16_t mtu_present;
|
|
||||||
uint16_t mtu;
|
|
||||||
uint16_t flush_to_present;
|
|
||||||
uint16_t flush_to;
|
|
||||||
uint16_t qos_present;
|
|
||||||
FLOW_SPEC qos;
|
|
||||||
uint16_t fcr_present;
|
|
||||||
tL2CAP_FCR fcr;
|
|
||||||
uint16_t fcs_present;
|
|
||||||
uint16_t fcs;
|
|
||||||
uint16_t ext_flow_spec_present;
|
|
||||||
FLOW_SPEC ext_flow_spec;
|
|
||||||
} tL2CAP_CFG_INFO;
|
|
||||||
|
|
||||||
// Basic L2CAP link control block
|
|
||||||
typedef struct {
|
|
||||||
bool wait_ack;
|
|
||||||
// Other FCR fields - not needed for our specific hook
|
|
||||||
} tL2C_FCRB;
|
|
||||||
|
|
||||||
// Forward declarations for needed types
|
|
||||||
struct t_l2c_rcb;
|
|
||||||
struct t_l2c_lcb;
|
|
||||||
|
|
||||||
typedef struct t_l2c_ccb {
|
|
||||||
struct t_l2c_ccb* p_next_ccb; // Next CCB in the chain
|
|
||||||
struct t_l2c_ccb* p_prev_ccb; // Previous CCB in the chain
|
|
||||||
struct t_l2c_lcb* p_lcb; // Link this CCB belongs to
|
|
||||||
struct t_l2c_rcb* p_rcb; // Registration CB for this Channel
|
|
||||||
uint16_t local_cid; // Local CID
|
|
||||||
uint16_t remote_cid; // Remote CID
|
|
||||||
uint16_t p_lcb_next; // For linking CCBs to an LCB
|
|
||||||
uint8_t ccb_priority; // Channel priority
|
|
||||||
uint16_t tx_mps; // MPS for outgoing messages
|
|
||||||
uint16_t max_rx_mtu; // Max MTU we will receive
|
|
||||||
// State variables
|
|
||||||
bool in_use; // True when channel active
|
|
||||||
uint8_t chnl_state; // Channel state
|
|
||||||
uint8_t local_id; // Transaction ID for local trans
|
|
||||||
uint8_t remote_id; // Transaction ID for remote
|
|
||||||
uint8_t timer_entry; // Timer entry
|
|
||||||
uint8_t is_flushable; // True if flushable
|
|
||||||
// Configuration variables
|
|
||||||
uint16_t our_cfg_bits; // Bitmap of local config bits
|
|
||||||
uint16_t peer_cfg_bits; // Bitmap of peer config bits
|
|
||||||
uint16_t config_done; // Configuration bitmask
|
|
||||||
uint16_t remote_config_rsp_result; // Remote config response result
|
|
||||||
tL2CAP_CFG_INFO our_cfg; // Our saved configuration options
|
|
||||||
tL2CAP_CFG_INFO peer_cfg; // Peer's saved configuration options
|
|
||||||
// Additional control fields
|
|
||||||
uint8_t remote_credit_count; // Credits sent to peer
|
|
||||||
tL2C_FCRB fcrb; // FCR info
|
|
||||||
bool ecoc; // Enhanced Credit-based mode
|
|
||||||
} tL2C_CCB;
|
|
||||||
|
|
||||||
static uint8_t (*original_l2c_fcr_chk_chan_modes)(void* p_ccb) = nullptr;
|
|
||||||
static void (*original_l2cu_process_our_cfg_req)(tL2C_CCB* p_ccb, tL2CAP_CFG_INFO* p_cfg) = nullptr;
|
|
||||||
static void (*original_l2c_csm_config)(tL2C_CCB* p_ccb, uint8_t event, void* p_data) = nullptr;
|
|
||||||
static void (*original_l2cu_send_peer_info_req)(tL2C_LCB* p_lcb, uint16_t info_type) = nullptr;
|
|
||||||
|
|
||||||
// Add original pointer for BTA_DmSetLocalDiRecord
|
|
||||||
static tBTA_STATUS (*original_BTA_DmSetLocalDiRecord)(tSDP_DI_RECORD* p_device_info, uint32_t* p_handle) = nullptr;
|
|
||||||
|
|
||||||
uint8_t fake_l2c_fcr_chk_chan_modes(void* p_ccb) {
|
uint8_t fake_l2c_fcr_chk_chan_modes(void* p_ccb) {
|
||||||
LOGI("l2c_fcr_chk_chan_modes hooked, returning true.");
|
LOGI("l2c_fcr_chk_chan_modes hooked");
|
||||||
|
uint8_t orig = 0;
|
||||||
|
if (original_l2c_fcr_chk_chan_modes)
|
||||||
|
orig = original_l2c_fcr_chk_chan_modes(p_ccb);
|
||||||
|
|
||||||
|
LOGI("Original returned %d, forcing 1", orig);
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
void fake_l2cu_process_our_cfg_req(tL2C_CCB* p_ccb, tL2CAP_CFG_INFO* p_cfg) {
|
tBTA_STATUS fake_BTA_DmSetLocalDiRecord(
|
||||||
original_l2cu_process_our_cfg_req(p_ccb, p_cfg);
|
tSDP_DI_RECORD* p_device_info,
|
||||||
p_ccb->our_cfg.fcr.mode = 0x00;
|
uint32_t* p_handle) {
|
||||||
LOGI("Set FCR mode to Basic Mode in outgoing config request");
|
|
||||||
}
|
|
||||||
|
|
||||||
void fake_l2c_csm_config(tL2C_CCB* p_ccb, uint8_t event, void* p_data) {
|
LOGI("BTA_DmSetLocalDiRecord hooked");
|
||||||
// Call the original function first to handle the specific code path where the FCR mode is checked
|
|
||||||
original_l2c_csm_config(p_ccb, event, p_data);
|
|
||||||
|
|
||||||
// Check if this happens during CONFIG_RSP event handling
|
|
||||||
if (event == L2CEVT_L2CAP_CONFIG_RSP) {
|
|
||||||
p_ccb->our_cfg.fcr.mode = p_ccb->peer_cfg.fcr.mode;
|
|
||||||
LOGI("Forced compatibility in l2c_csm_config: set our_mode=%d to match peer_mode=%d",
|
|
||||||
p_ccb->our_cfg.fcr.mode, p_ccb->peer_cfg.fcr.mode);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Replacement function that does nothing
|
|
||||||
void fake_l2cu_send_peer_info_req(tL2C_LCB* p_lcb, uint16_t info_type) {
|
|
||||||
LOGI("Intercepted l2cu_send_peer_info_req for info_type 0x%04x - doing nothing", info_type);
|
|
||||||
// Just return without doing anything
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// New loader for SDP hook offset (persist.librepods.sdp_offset)
|
|
||||||
uintptr_t loadSdpOffset() {
|
|
||||||
const char* property_name = "persist.librepods.sdp_offset";
|
|
||||||
char value[PROP_VALUE_MAX] = {0};
|
|
||||||
|
|
||||||
int len = __system_property_get(property_name, value);
|
|
||||||
if (len > 0) {
|
|
||||||
LOGI("Read sdp offset from property: %s", value);
|
|
||||||
uintptr_t offset;
|
|
||||||
char* endptr = nullptr;
|
|
||||||
|
|
||||||
const char* parse_start = value;
|
|
||||||
if (value[0] == '0' && (value[1] == 'x' || value[1] == 'X')) {
|
|
||||||
parse_start = value + 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
errno = 0;
|
|
||||||
offset = strtoul(parse_start, &endptr, 16);
|
|
||||||
|
|
||||||
if (errno == 0 && endptr != parse_start && *endptr == '\0' && offset > 0) {
|
|
||||||
LOGI("Parsed sdp offset: 0x%x", offset);
|
|
||||||
return offset;
|
|
||||||
}
|
|
||||||
|
|
||||||
LOGE("Failed to parse sdp offset from property value: %s", value);
|
|
||||||
}
|
|
||||||
|
|
||||||
LOGI("No sdp offset property present - skipping SDP hook");
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Fake BTA_DmSetLocalDiRecord: set vendor/vendor_id_source then call original
|
|
||||||
tBTA_STATUS fake_BTA_DmSetLocalDiRecord(tSDP_DI_RECORD* p_device_info, uint32_t* p_handle) {
|
|
||||||
LOGI("BTA_DmSetLocalDiRecord hooked - forcing vendor fields");
|
|
||||||
if (p_device_info) {
|
if (p_device_info) {
|
||||||
p_device_info->vendor = 0x004C;
|
p_device_info->vendor = 0x004C;
|
||||||
p_device_info->vendor_id_source = 0x0001;
|
p_device_info->vendor_id_source = 0x0001;
|
||||||
}
|
}
|
||||||
LOGI("Set vendor=0x%04x, vendor_id_source=0x%04x", p_device_info->vendor, p_device_info->vendor_id_source);
|
|
||||||
if (original_BTA_DmSetLocalDiRecord) {
|
|
||||||
return original_BTA_DmSetLocalDiRecord(p_device_info, p_handle);
|
|
||||||
}
|
|
||||||
|
|
||||||
LOGE("Original BTA_DmSetLocalDiRecord not available");
|
if (original_BTA_DmSetLocalDiRecord)
|
||||||
|
return original_BTA_DmSetLocalDiRecord(p_device_info, p_handle);
|
||||||
|
|
||||||
return BTA_FAILURE;
|
return BTA_FAILURE;
|
||||||
}
|
}
|
||||||
|
|
||||||
uintptr_t loadHookOffset([[maybe_unused]] const char* package_name) {
|
static bool decompressXZ(
|
||||||
const char* property_name = "persist.librepods.hook_offset";
|
const uint8_t* input,
|
||||||
char value[PROP_VALUE_MAX] = {0};
|
size_t input_size,
|
||||||
|
std::vector<uint8_t>& output) {
|
||||||
|
|
||||||
int len = __system_property_get(property_name, value);
|
xz_crc32_init();
|
||||||
if (len > 0) {
|
#ifdef XZ_USE_CRC64
|
||||||
LOGI("Read hook offset from property: %s", value);
|
xz_crc64_init();
|
||||||
uintptr_t offset;
|
#endif
|
||||||
char* endptr = nullptr;
|
|
||||||
|
|
||||||
const char* parse_start = value;
|
struct xz_dec* dec = xz_dec_init(XZ_DYNALLOC, 64U << 20);
|
||||||
if (value[0] == '0' && (value[1] == 'x' || value[1] == 'X')) {
|
if (!dec) return false;
|
||||||
parse_start = value + 2;
|
|
||||||
|
struct xz_buf buf{};
|
||||||
|
buf.in = input;
|
||||||
|
buf.in_pos = 0;
|
||||||
|
buf.in_size = input_size;
|
||||||
|
|
||||||
|
output.resize(input_size * 8);
|
||||||
|
|
||||||
|
buf.out = output.data();
|
||||||
|
buf.out_pos = 0;
|
||||||
|
buf.out_size = output.size();
|
||||||
|
|
||||||
|
while (true) {
|
||||||
|
enum xz_ret ret = xz_dec_run(dec, &buf);
|
||||||
|
|
||||||
|
if (ret == XZ_STREAM_END)
|
||||||
|
break;
|
||||||
|
|
||||||
|
if (ret != XZ_OK) {
|
||||||
|
xz_dec_end(dec);
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
errno = 0;
|
if (buf.out_pos == buf.out_size) {
|
||||||
offset = strtoul(parse_start, &endptr, 16);
|
size_t old = output.size();
|
||||||
|
output.resize(old * 2);
|
||||||
if (errno == 0 && endptr != parse_start && *endptr == '\0' && offset > 0) {
|
buf.out = output.data();
|
||||||
LOGI("Parsed offset: 0x%x", offset);
|
buf.out_size = output.size();
|
||||||
return offset;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
LOGE("Failed to parse offset from property value: %s", value);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
LOGI("Using hardcoded fallback offset");
|
output.resize(buf.out_pos);
|
||||||
return 0x00a55e30;
|
xz_dec_end(dec);
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
uintptr_t loadL2cuProcessCfgReqOffset() {
|
static bool getLibraryPath(const char* name, std::string& out) {
|
||||||
const char* property_name = "persist.librepods.cfg_req_offset";
|
FILE* fp = fopen("/proc/self/maps", "r");
|
||||||
char value[PROP_VALUE_MAX] = {0};
|
if (!fp) return false;
|
||||||
|
|
||||||
int len = __system_property_get(property_name, value);
|
|
||||||
if (len > 0) {
|
|
||||||
LOGI("Read l2cu_process_our_cfg_req offset from property: %s", value);
|
|
||||||
uintptr_t offset;
|
|
||||||
char* endptr = nullptr;
|
|
||||||
|
|
||||||
const char* parse_start = value;
|
|
||||||
if (value[0] == '0' && (value[1] == 'x' || value[1] == 'X')) {
|
|
||||||
parse_start = value + 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
errno = 0;
|
|
||||||
offset = strtoul(parse_start, &endptr, 16);
|
|
||||||
|
|
||||||
if (errno == 0 && endptr != parse_start && *endptr == '\0' && offset > 0) {
|
|
||||||
LOGI("Parsed l2cu_process_our_cfg_req offset: 0x%x", offset);
|
|
||||||
return offset;
|
|
||||||
}
|
|
||||||
|
|
||||||
LOGE("Failed to parse l2cu_process_our_cfg_req offset from property value: %s", value);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Return 0 if not found - we'll skip this hook
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
uintptr_t loadL2cCsmConfigOffset() {
|
|
||||||
const char* property_name = "persist.librepods.csm_config_offset";
|
|
||||||
char value[PROP_VALUE_MAX] = {0};
|
|
||||||
|
|
||||||
int len = __system_property_get(property_name, value);
|
|
||||||
if (len > 0) {
|
|
||||||
LOGI("Read l2c_csm_config offset from property: %s", value);
|
|
||||||
uintptr_t offset;
|
|
||||||
char* endptr = nullptr;
|
|
||||||
|
|
||||||
const char* parse_start = value;
|
|
||||||
if (value[0] == '0' && (value[1] == 'x' || value[1] == 'X')) {
|
|
||||||
parse_start = value + 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
errno = 0;
|
|
||||||
offset = strtoul(parse_start, &endptr, 16);
|
|
||||||
|
|
||||||
if (errno == 0 && endptr != parse_start && *endptr == '\0' && offset > 0) {
|
|
||||||
LOGI("Parsed l2c_csm_config offset: 0x%x", offset);
|
|
||||||
return offset;
|
|
||||||
}
|
|
||||||
|
|
||||||
LOGE("Failed to parse l2c_csm_config offset from property value: %s", value);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Return 0 if not found - we'll skip this hook
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
uintptr_t loadL2cuSendPeerInfoReqOffset() {
|
|
||||||
const char* property_name = "persist.librepods.peer_info_req_offset";
|
|
||||||
char value[PROP_VALUE_MAX] = {0};
|
|
||||||
|
|
||||||
int len = __system_property_get(property_name, value);
|
|
||||||
if (len > 0) {
|
|
||||||
LOGI("Read l2cu_send_peer_info_req offset from property: %s", value);
|
|
||||||
uintptr_t offset;
|
|
||||||
char* endptr = nullptr;
|
|
||||||
|
|
||||||
const char* parse_start = value;
|
|
||||||
if (value[0] == '0' && (value[1] == 'x' || value[1] == 'X')) {
|
|
||||||
parse_start = value + 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
errno = 0;
|
|
||||||
offset = strtoul(parse_start, &endptr, 16);
|
|
||||||
|
|
||||||
if (errno == 0 && endptr != parse_start && *endptr == '\0' && offset > 0) {
|
|
||||||
LOGI("Parsed l2cu_send_peer_info_req offset: 0x%x", offset);
|
|
||||||
return offset;
|
|
||||||
}
|
|
||||||
|
|
||||||
LOGE("Failed to parse l2cu_send_peer_info_req offset from property value: %s", value);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Return 0 if not found - we'll skip this hook
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
uintptr_t getModuleBase(const char *module_name) {
|
|
||||||
FILE *fp;
|
|
||||||
char line[1024];
|
char line[1024];
|
||||||
uintptr_t base_addr = 0;
|
|
||||||
|
|
||||||
fp = fopen("/proc/self/maps", "r");
|
|
||||||
if (!fp) {
|
|
||||||
LOGE("Failed to open /proc/self/maps");
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
while (fgets(line, sizeof(line), fp)) {
|
while (fgets(line, sizeof(line), fp)) {
|
||||||
if (strstr(line, module_name)) {
|
if (strstr(line, name)) {
|
||||||
char *start_addr_str = line;
|
char* path = strchr(line, '/');
|
||||||
char *end_addr_str = strchr(line, '-');
|
if (path) {
|
||||||
if (end_addr_str) {
|
out = path;
|
||||||
*end_addr_str = '\0';
|
out.erase(out.find('\n'));
|
||||||
base_addr = strtoull(start_addr_str, nullptr, 16);
|
fclose(fp);
|
||||||
break;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fclose(fp);
|
fclose(fp);
|
||||||
return base_addr;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool findAndHookFunction(const char *library_name) {
|
static uintptr_t getModuleBase(const char* name) {
|
||||||
|
FILE* fp = fopen("/proc/self/maps", "r");
|
||||||
|
if (!fp) return 0;
|
||||||
|
|
||||||
|
char line[1024];
|
||||||
|
uintptr_t base = 0;
|
||||||
|
|
||||||
|
while (fgets(line, sizeof(line), fp)) {
|
||||||
|
if (strstr(line, name)) {
|
||||||
|
base = strtoull(line, nullptr, 16);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fclose(fp);
|
||||||
|
return base;
|
||||||
|
}
|
||||||
|
|
||||||
|
static uint64_t findSymbolOffset(
|
||||||
|
const std::vector<uint8_t>& elf,
|
||||||
|
const char* symbol_substring) {
|
||||||
|
|
||||||
|
auto* eh = reinterpret_cast<const Elf64_Ehdr*>(elf.data());
|
||||||
|
auto* shdr = reinterpret_cast<const Elf64_Shdr*>(
|
||||||
|
elf.data() + eh->e_shoff);
|
||||||
|
|
||||||
|
const char* shstr =
|
||||||
|
reinterpret_cast<const char*>(
|
||||||
|
elf.data() + shdr[eh->e_shstrndx].sh_offset);
|
||||||
|
|
||||||
|
const Elf64_Shdr* symtab = nullptr;
|
||||||
|
const Elf64_Shdr* strtab = nullptr;
|
||||||
|
|
||||||
|
for (int i = 0; i < eh->e_shnum; ++i) {
|
||||||
|
const char* secname = shstr + shdr[i].sh_name;
|
||||||
|
if (!strcmp(secname, ".symtab"))
|
||||||
|
symtab = &shdr[i];
|
||||||
|
if (!strcmp(secname, ".strtab"))
|
||||||
|
strtab = &shdr[i];
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!symtab || !strtab)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
auto* symbols = reinterpret_cast<const Elf64_Sym*>(
|
||||||
|
elf.data() + symtab->sh_offset);
|
||||||
|
|
||||||
|
const char* strings =
|
||||||
|
reinterpret_cast<const char*>(
|
||||||
|
elf.data() + strtab->sh_offset);
|
||||||
|
|
||||||
|
size_t count = symtab->sh_size / sizeof(Elf64_Sym);
|
||||||
|
|
||||||
|
for (size_t i = 0; i < count; ++i) {
|
||||||
|
const char* name = strings + symbols[i].st_name;
|
||||||
|
|
||||||
|
if (strstr(name, symbol_substring) &&
|
||||||
|
ELF64_ST_TYPE(symbols[i].st_info) == STT_FUNC) {
|
||||||
|
|
||||||
|
LOGI("Resolved %s at 0x%lx",
|
||||||
|
name,
|
||||||
|
(unsigned long)symbols[i].st_value);
|
||||||
|
|
||||||
|
return symbols[i].st_value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool hookLibrary(const char* libname) {
|
||||||
|
|
||||||
if (!hook_func) {
|
if (!hook_func) {
|
||||||
LOGE("Hook function not initialized");
|
LOGE("hook_func not initialized");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
uintptr_t base_addr = getModuleBase(library_name);
|
std::string path;
|
||||||
if (!base_addr) {
|
if (!getLibraryPath(libname, path)) {
|
||||||
LOGE("Failed to get base address of %s", library_name);
|
LOGE("Failed to locate %s", libname);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Load all offsets from system properties - no hardcoding
|
int fd = open(path.c_str(), O_RDONLY);
|
||||||
uintptr_t l2c_fcr_offset = loadHookOffset(nullptr);
|
if (fd < 0) return false;
|
||||||
uintptr_t l2cu_process_our_cfg_req_offset = loadL2cuProcessCfgReqOffset();
|
|
||||||
uintptr_t l2c_csm_config_offset = loadL2cCsmConfigOffset();
|
|
||||||
uintptr_t l2cu_send_peer_info_req_offset = loadL2cuSendPeerInfoReqOffset();
|
|
||||||
uintptr_t sdp_offset = loadSdpOffset();
|
|
||||||
|
|
||||||
bool success = false;
|
struct stat st{};
|
||||||
|
if (fstat(fd, &st) != 0) {
|
||||||
// Hook l2c_fcr_chk_chan_modes - this is our primary hook
|
close(fd);
|
||||||
if (l2c_fcr_offset > 0) {
|
|
||||||
void* target = reinterpret_cast<void*>(base_addr + l2c_fcr_offset);
|
|
||||||
LOGI("Hooking l2c_fcr_chk_chan_modes at offset: 0x%x, base: %p, target: %p",
|
|
||||||
l2c_fcr_offset, (void*)base_addr, target);
|
|
||||||
|
|
||||||
int result = hook_func(target, (void*)fake_l2c_fcr_chk_chan_modes, (void**)&original_l2c_fcr_chk_chan_modes);
|
|
||||||
if (result != 0) {
|
|
||||||
LOGE("Failed to hook l2c_fcr_chk_chan_modes, error: %d", result);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
LOGI("Successfully hooked l2c_fcr_chk_chan_modes");
|
|
||||||
success = true;
|
|
||||||
} else {
|
|
||||||
LOGE("No valid offset for l2c_fcr_chk_chan_modes found, cannot proceed");
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Hook l2cu_process_our_cfg_req if offset is available
|
std::vector<uint8_t> file(st.st_size);
|
||||||
if (l2cu_process_our_cfg_req_offset > 0) {
|
read(fd, file.data(), st.st_size);
|
||||||
void* target = reinterpret_cast<void*>(base_addr + l2cu_process_our_cfg_req_offset);
|
close(fd);
|
||||||
LOGI("Hooking l2cu_process_our_cfg_req at offset: 0x%x, base: %p, target: %p",
|
|
||||||
l2cu_process_our_cfg_req_offset, (void*)base_addr, target);
|
|
||||||
|
|
||||||
int result = hook_func(target, (void*)fake_l2cu_process_our_cfg_req, (void**)&original_l2cu_process_our_cfg_req);
|
auto* eh = reinterpret_cast<Elf64_Ehdr*>(file.data());
|
||||||
if (result != 0) {
|
auto* shdr = reinterpret_cast<Elf64_Shdr*>(
|
||||||
LOGE("Failed to hook l2cu_process_our_cfg_req, error: %d", result);
|
file.data() + eh->e_shoff);
|
||||||
// Continue even if this hook fails
|
|
||||||
} else {
|
const char* shstr =
|
||||||
LOGI("Successfully hooked l2cu_process_our_cfg_req");
|
reinterpret_cast<const char*>(
|
||||||
|
file.data() + shdr[eh->e_shstrndx].sh_offset);
|
||||||
|
|
||||||
|
for (int i = 0; i < eh->e_shnum; ++i) {
|
||||||
|
|
||||||
|
if (!strcmp(shstr + shdr[i].sh_name, ".gnu_debugdata")) {
|
||||||
|
|
||||||
|
std::vector<uint8_t> compressed(
|
||||||
|
file.begin() + shdr[i].sh_offset,
|
||||||
|
file.begin() + shdr[i].sh_offset + shdr[i].sh_size);
|
||||||
|
|
||||||
|
std::vector<uint8_t> decompressed;
|
||||||
|
|
||||||
|
if (!decompressXZ(
|
||||||
|
compressed.data(),
|
||||||
|
compressed.size(),
|
||||||
|
decompressed))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
uintptr_t base = getModuleBase(libname);
|
||||||
|
if (!base) return false;
|
||||||
|
|
||||||
|
uint64_t chk_offset =
|
||||||
|
findSymbolOffset(decompressed,
|
||||||
|
"l2c_fcr_chk_chan_modes");
|
||||||
|
|
||||||
|
// uint64_t sdp_offset =
|
||||||
|
// findSymbolOffset(decompressed,
|
||||||
|
// "BTA_DmSetLocalDiRecord");
|
||||||
|
|
||||||
|
if (chk_offset) {
|
||||||
|
void* target =
|
||||||
|
reinterpret_cast<void*>(base + chk_offset);
|
||||||
|
|
||||||
|
hook_func(target,
|
||||||
|
(void*)fake_l2c_fcr_chk_chan_modes,
|
||||||
|
(void**)&original_l2c_fcr_chk_chan_modes);
|
||||||
|
|
||||||
|
LOGI("Hooked l2c_fcr_chk_chan_modes");
|
||||||
|
}
|
||||||
|
|
||||||
|
// if (sdp_offset) {
|
||||||
|
// void* target =
|
||||||
|
// reinterpret_cast<void*>(base + sdp_offset);
|
||||||
|
//
|
||||||
|
// hook_func(target,
|
||||||
|
// (void*)fake_BTA_DmSetLocalDiRecord,
|
||||||
|
// (void**)&original_BTA_DmSetLocalDiRecord);
|
||||||
|
//
|
||||||
|
// LOGI("Hooked BTA_DmSetLocalDiRecord");
|
||||||
|
// }
|
||||||
|
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
LOGI("Skipping l2cu_process_our_cfg_req hook as offset is not available");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Hook l2c_csm_config if offset is available
|
return false;
|
||||||
if (l2c_csm_config_offset > 0) {
|
|
||||||
void* target = reinterpret_cast<void*>(base_addr + l2c_csm_config_offset);
|
|
||||||
LOGI("Hooking l2c_csm_config at offset: 0x%x, base: %p, target: %p",
|
|
||||||
l2c_csm_config_offset, (void*)base_addr, target);
|
|
||||||
|
|
||||||
int result = hook_func(target, (void*)fake_l2c_csm_config, (void**)&original_l2c_csm_config);
|
|
||||||
if (result != 0) {
|
|
||||||
LOGE("Failed to hook l2c_csm_config, error: %d", result);
|
|
||||||
// Continue even if this hook fails
|
|
||||||
} else {
|
|
||||||
LOGI("Successfully hooked l2c_csm_config");
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
LOGI("Skipping l2c_csm_config hook as offset is not available");
|
|
||||||
}
|
|
||||||
|
|
||||||
// Hook l2cu_send_peer_info_req if offset is available
|
|
||||||
if (l2cu_send_peer_info_req_offset > 0) {
|
|
||||||
void* target = reinterpret_cast<void*>(base_addr + l2cu_send_peer_info_req_offset);
|
|
||||||
LOGI("Hooking l2cu_send_peer_info_req at offset: 0x%x, base: %p, target: %p",
|
|
||||||
l2cu_send_peer_info_req_offset, (void*)base_addr, target);
|
|
||||||
|
|
||||||
int result = hook_func(target, (void*)fake_l2cu_send_peer_info_req, (void**)&original_l2cu_send_peer_info_req);
|
|
||||||
if (result != 0) {
|
|
||||||
LOGE("Failed to hook l2cu_send_peer_info_req, error: %d", result);
|
|
||||||
// Continue even if this hook fails
|
|
||||||
} else {
|
|
||||||
LOGI("Successfully hooked l2cu_send_peer_info_req");
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
LOGI("Skipping l2cu_send_peer_info_req hook as offset is not available");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (sdp_offset > 0) {
|
|
||||||
void* target = reinterpret_cast<void*>(base_addr + sdp_offset);
|
|
||||||
LOGI("Hooking BTA_DmSetLocalDiRecord at offset: 0x%x, base: %p, target: %p",
|
|
||||||
sdp_offset, (void*)base_addr, target);
|
|
||||||
|
|
||||||
int result = hook_func(target, (void*)fake_BTA_DmSetLocalDiRecord, (void**)&original_BTA_DmSetLocalDiRecord);
|
|
||||||
if (result != 0) {
|
|
||||||
LOGE("Failed to hook BTA_DmSetLocalDiRecord, error: %d", result);
|
|
||||||
} else {
|
|
||||||
LOGI("Successfully hooked BTA_DmSetLocalDiRecord (SDP)");
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
LOGI("Skipping BTA_DmSetLocalDiRecord hook as sdp offset is not available");
|
|
||||||
}
|
|
||||||
|
|
||||||
return success;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void on_library_loaded(const char *name, [[maybe_unused]] void *handle) {
|
static void on_library_loaded(const char* name, void*) {
|
||||||
|
|
||||||
if (strstr(name, "libbluetooth_jni.so")) {
|
if (strstr(name, "libbluetooth_jni.so")) {
|
||||||
LOGI("Detected Bluetooth JNI library: %s", name);
|
LOGI("Bluetooth JNI loaded");
|
||||||
|
hookLibrary("libbluetooth_jni.so");
|
||||||
|
}
|
||||||
|
|
||||||
bool hooked = findAndHookFunction("libbluetooth_jni.so");
|
if (strstr(name, "libbluetooth_qti.so")) {
|
||||||
if (!hooked) {
|
LOGI("Bluetooth QTI loaded");
|
||||||
LOGE("Failed to hook Bluetooth JNI library function");
|
hookLibrary("libbluetooth_qti.so");
|
||||||
}
|
|
||||||
} else if (strstr(name, "libbluetooth_qti.so")) {
|
|
||||||
LOGI("Detected Bluetooth QTI library: %s", name);
|
|
||||||
|
|
||||||
bool hooked = findAndHookFunction("libbluetooth_qti.so");
|
|
||||||
if (!hooked) {
|
|
||||||
LOGE("Failed to hook Bluetooth QTI library function");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
extern "C" [[gnu::visibility("default")]] [[gnu::used]]
|
extern "C"
|
||||||
|
[[gnu::visibility("default")]]
|
||||||
|
[[gnu::used]]
|
||||||
NativeOnModuleLoaded native_init(const NativeAPIEntries* entries) {
|
NativeOnModuleLoaded native_init(const NativeAPIEntries* entries) {
|
||||||
LOGI("L2C FCR Hook module initialized");
|
|
||||||
|
|
||||||
hook_func = entries->hook_func;
|
LOGI("LibrePods initialized");
|
||||||
|
|
||||||
|
hook_func = (HookFunType)entries->hook_func;
|
||||||
|
|
||||||
return on_library_loaded;
|
return on_library_loaded;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,50 +1,52 @@
|
|||||||
#pragma once
|
/*
|
||||||
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
|
Copyright (C) 2025 LibrePods contributors
|
||||||
|
|
||||||
|
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
|
||||||
|
any later version.
|
||||||
|
|
||||||
|
This program is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
GNU General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU General Public License
|
||||||
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
#include <cstdint>
|
#include <cstdint>
|
||||||
#include <vector>
|
|
||||||
|
|
||||||
typedef int (*HookFunType)(void *func, void *replace, void **backup);
|
typedef int (*HookFunType)(void *func, void *replace, void **backup);
|
||||||
|
|
||||||
typedef int (*UnhookFunType)(void *func);
|
|
||||||
|
|
||||||
typedef void (*NativeOnModuleLoaded)(const char *name, void *handle);
|
typedef void (*NativeOnModuleLoaded)(const char *name, void *handle);
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
uint32_t version;
|
uint32_t version;
|
||||||
HookFunType hook_func;
|
void* hook_func;
|
||||||
UnhookFunType unhook_func;
|
void* unhook_func;
|
||||||
} NativeAPIEntries;
|
} NativeAPIEntries;
|
||||||
|
|
||||||
[[maybe_unused]] typedef NativeOnModuleLoaded (*NativeInit)(const NativeAPIEntries *entries);
|
typedef NativeOnModuleLoaded (*NativeInit)(const NativeAPIEntries *entries);
|
||||||
|
|
||||||
typedef struct t_l2c_ccb tL2C_CCB;
|
|
||||||
typedef struct t_l2c_lcb tL2C_LCB;
|
|
||||||
|
|
||||||
uintptr_t loadHookOffset(const char* package_name);
|
|
||||||
uintptr_t getModuleBase(const char *module_name);
|
|
||||||
uintptr_t loadL2cuProcessCfgReqOffset();
|
|
||||||
uintptr_t loadL2cCsmConfigOffset();
|
|
||||||
uintptr_t loadL2cuSendPeerInfoReqOffset();
|
|
||||||
bool findAndHookFunction(const char *library_path);
|
|
||||||
|
|
||||||
#define SDP_MAX_ATTR_LEN 400
|
|
||||||
|
|
||||||
typedef struct t_sdp_di_record {
|
|
||||||
uint16_t vendor;
|
|
||||||
uint16_t vendor_id_source;
|
|
||||||
uint16_t product;
|
|
||||||
uint16_t version;
|
|
||||||
bool primary_record;
|
|
||||||
char client_executable_url[SDP_MAX_ATTR_LEN];
|
|
||||||
char service_description[SDP_MAX_ATTR_LEN];
|
|
||||||
char documentation_url[SDP_MAX_ATTR_LEN];
|
|
||||||
} tSDP_DI_RECORD;
|
|
||||||
|
|
||||||
typedef enum : uint8_t {
|
typedef enum : uint8_t {
|
||||||
BTA_SUCCESS = 0, /* Successful operation. */
|
BTA_SUCCESS = 0, /* Successful operation. */
|
||||||
BTA_FAILURE = 1, /* Generic failure. */
|
BTA_FAILURE = 1, /* Generic failure. */
|
||||||
BTA_PENDING = 2, /* API cannot be completed right now */
|
BTA_PENDING = 2, /* API cannot be completed right now */
|
||||||
BTA_BUSY = 3,
|
BTA_BUSY = 3,
|
||||||
BTA_NO_RESOURCES = 4,
|
BTA_NO_RESOURCES = 4,
|
||||||
BTA_WRONG_MODE = 5,
|
BTA_WRONG_MODE = 5,
|
||||||
} tBTA_STATUS;
|
} tBTA_STATUS;
|
||||||
|
|
||||||
|
typedef struct t_sdp_di_record {
|
||||||
|
uint16_t vendor;
|
||||||
|
uint16_t vendor_id_source;
|
||||||
|
uint16_t product;
|
||||||
|
uint16_t version;
|
||||||
|
bool primary_record;
|
||||||
|
char client_executable_url[400];
|
||||||
|
char service_description[400];
|
||||||
|
char documentation_url[400];
|
||||||
|
} tSDP_DI_RECORD;
|
||||||
|
|||||||
448
android/app/src/main/cpp/xz/xz.h
Normal file
448
android/app/src/main/cpp/xz/xz.h
Normal file
@@ -0,0 +1,448 @@
|
|||||||
|
/* SPDX-License-Identifier: 0BSD */
|
||||||
|
|
||||||
|
/*
|
||||||
|
* XZ decompressor
|
||||||
|
*
|
||||||
|
* Authors: Lasse Collin <lasse.collin@tukaani.org>
|
||||||
|
* Igor Pavlov <https://7-zip.org/>
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef XZ_H
|
||||||
|
#define XZ_H
|
||||||
|
|
||||||
|
#ifdef __KERNEL__
|
||||||
|
# include <linux/stddef.h>
|
||||||
|
# include <linux/types.h>
|
||||||
|
#else
|
||||||
|
# include <stddef.h>
|
||||||
|
# include <stdint.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
extern "C" {
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* "#define XZ_EXTERN static" can be used to make extern functions static. */
|
||||||
|
#ifndef XZ_EXTERN
|
||||||
|
# define XZ_EXTERN extern
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/**
|
||||||
|
* enum xz_mode - Operation mode
|
||||||
|
*
|
||||||
|
* @XZ_SINGLE: Single-call mode. This uses less RAM than
|
||||||
|
* multi-call modes, because the LZMA2
|
||||||
|
* dictionary doesn't need to be allocated as
|
||||||
|
* part of the decoder state. All required data
|
||||||
|
* structures are allocated at initialization,
|
||||||
|
* so xz_dec_run() cannot return XZ_MEM_ERROR.
|
||||||
|
* @XZ_PREALLOC: Multi-call mode with preallocated LZMA2
|
||||||
|
* dictionary buffer. All data structures are
|
||||||
|
* allocated at initialization, so xz_dec_run()
|
||||||
|
* cannot return XZ_MEM_ERROR.
|
||||||
|
* @XZ_DYNALLOC: Multi-call mode. The LZMA2 dictionary is
|
||||||
|
* allocated once the required size has been
|
||||||
|
* parsed from the stream headers. If the
|
||||||
|
* allocation fails, xz_dec_run() will return
|
||||||
|
* XZ_MEM_ERROR.
|
||||||
|
*
|
||||||
|
* It is possible to enable support only for a subset of the above
|
||||||
|
* modes at compile time by defining XZ_DEC_SINGLE, XZ_DEC_PREALLOC,
|
||||||
|
* or XZ_DEC_DYNALLOC. The xz_dec kernel module is always compiled
|
||||||
|
* with support for all operation modes, but the preboot code may
|
||||||
|
* be built with fewer features to minimize code size.
|
||||||
|
*/
|
||||||
|
enum xz_mode {
|
||||||
|
XZ_SINGLE,
|
||||||
|
XZ_PREALLOC,
|
||||||
|
XZ_DYNALLOC
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* enum xz_ret - Return codes
|
||||||
|
* @XZ_OK: Everything is OK so far. More input or more
|
||||||
|
* output space is required to continue. This
|
||||||
|
* return code is possible only in multi-call mode
|
||||||
|
* (XZ_PREALLOC or XZ_DYNALLOC).
|
||||||
|
* @XZ_STREAM_END: Operation finished successfully.
|
||||||
|
* @XZ_UNSUPPORTED_CHECK: Integrity check type is not supported. Decoding
|
||||||
|
* is still possible in multi-call mode by simply
|
||||||
|
* calling xz_dec_run() again.
|
||||||
|
* Note that this return value is used only if
|
||||||
|
* XZ_DEC_ANY_CHECK was defined at build time,
|
||||||
|
* which is not used in the kernel. Unsupported
|
||||||
|
* check types return XZ_OPTIONS_ERROR if
|
||||||
|
* XZ_DEC_ANY_CHECK was not defined at build time.
|
||||||
|
* @XZ_MEM_ERROR: Allocating memory failed. This return code is
|
||||||
|
* possible only if the decoder was initialized
|
||||||
|
* with XZ_DYNALLOC. The amount of memory that was
|
||||||
|
* tried to be allocated was no more than the
|
||||||
|
* dict_max argument given to xz_dec_init().
|
||||||
|
* @XZ_MEMLIMIT_ERROR: A bigger LZMA2 dictionary would be needed than
|
||||||
|
* allowed by the dict_max argument given to
|
||||||
|
* xz_dec_init(). This return value is possible
|
||||||
|
* only in multi-call mode (XZ_PREALLOC or
|
||||||
|
* XZ_DYNALLOC); the single-call mode (XZ_SINGLE)
|
||||||
|
* ignores the dict_max argument.
|
||||||
|
* @XZ_FORMAT_ERROR: File format was not recognized (wrong magic
|
||||||
|
* bytes).
|
||||||
|
* @XZ_OPTIONS_ERROR: This implementation doesn't support the requested
|
||||||
|
* compression options. In the decoder this means
|
||||||
|
* that the header CRC32 matches, but the header
|
||||||
|
* itself specifies something that we don't support.
|
||||||
|
* @XZ_DATA_ERROR: Compressed data is corrupt.
|
||||||
|
* @XZ_BUF_ERROR: Cannot make any progress. Details are slightly
|
||||||
|
* different between multi-call and single-call
|
||||||
|
* mode; more information below.
|
||||||
|
*
|
||||||
|
* In multi-call mode, XZ_BUF_ERROR is returned when two consecutive calls
|
||||||
|
* to XZ code cannot consume any input and cannot produce any new output.
|
||||||
|
* This happens when there is no new input available, or the output buffer
|
||||||
|
* is full while at least one output byte is still pending. Assuming your
|
||||||
|
* code is not buggy, you can get this error only when decoding a compressed
|
||||||
|
* stream that is truncated or otherwise corrupt.
|
||||||
|
*
|
||||||
|
* In single-call mode, XZ_BUF_ERROR is returned only when the output buffer
|
||||||
|
* is too small or the compressed input is corrupt in a way that makes the
|
||||||
|
* decoder produce more output than the caller expected. When it is
|
||||||
|
* (relatively) clear that the compressed input is truncated, XZ_DATA_ERROR
|
||||||
|
* is used instead of XZ_BUF_ERROR.
|
||||||
|
*/
|
||||||
|
enum xz_ret {
|
||||||
|
XZ_OK,
|
||||||
|
XZ_STREAM_END,
|
||||||
|
XZ_UNSUPPORTED_CHECK,
|
||||||
|
XZ_MEM_ERROR,
|
||||||
|
XZ_MEMLIMIT_ERROR,
|
||||||
|
XZ_FORMAT_ERROR,
|
||||||
|
XZ_OPTIONS_ERROR,
|
||||||
|
XZ_DATA_ERROR,
|
||||||
|
XZ_BUF_ERROR
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* struct xz_buf - Passing input and output buffers to XZ code
|
||||||
|
* @in: Beginning of the input buffer. This may be NULL if and only
|
||||||
|
* if in_pos is equal to in_size.
|
||||||
|
* @in_pos: Current position in the input buffer. This must not exceed
|
||||||
|
* in_size.
|
||||||
|
* @in_size: Size of the input buffer
|
||||||
|
* @out: Beginning of the output buffer. This may be NULL if and only
|
||||||
|
* if out_pos is equal to out_size.
|
||||||
|
* @out_pos: Current position in the output buffer. This must not exceed
|
||||||
|
* out_size.
|
||||||
|
* @out_size: Size of the output buffer
|
||||||
|
*
|
||||||
|
* Only the contents of the output buffer from out[out_pos] onward, and
|
||||||
|
* the variables in_pos and out_pos are modified by the XZ code.
|
||||||
|
*/
|
||||||
|
struct xz_buf {
|
||||||
|
const uint8_t *in;
|
||||||
|
size_t in_pos;
|
||||||
|
size_t in_size;
|
||||||
|
|
||||||
|
uint8_t *out;
|
||||||
|
size_t out_pos;
|
||||||
|
size_t out_size;
|
||||||
|
};
|
||||||
|
|
||||||
|
/*
|
||||||
|
* struct xz_dec - Opaque type to hold the XZ decoder state
|
||||||
|
*/
|
||||||
|
struct xz_dec;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* xz_dec_init() - Allocate and initialize a XZ decoder state
|
||||||
|
* @mode: Operation mode
|
||||||
|
* @dict_max: Maximum size of the LZMA2 dictionary (history buffer) for
|
||||||
|
* multi-call decoding. This is ignored in single-call mode
|
||||||
|
* (mode == XZ_SINGLE). LZMA2 dictionary is always 2^n bytes
|
||||||
|
* or 2^n + 2^(n-1) bytes (the latter sizes are less common
|
||||||
|
* in practice), so other values for dict_max don't make sense.
|
||||||
|
* In the kernel, dictionary sizes of 64 KiB, 128 KiB, 256 KiB,
|
||||||
|
* 512 KiB, and 1 MiB are probably the only reasonable values,
|
||||||
|
* except for kernel and initramfs images where a bigger
|
||||||
|
* dictionary can be fine and useful.
|
||||||
|
*
|
||||||
|
* Single-call mode (XZ_SINGLE): xz_dec_run() decodes the whole stream at
|
||||||
|
* once. The caller must provide enough output space or the decoding will
|
||||||
|
* fail. The output space is used as the dictionary buffer, which is why
|
||||||
|
* there is no need to allocate the dictionary as part of the decoder's
|
||||||
|
* internal state.
|
||||||
|
*
|
||||||
|
* Because the output buffer is used as the workspace, streams encoded using
|
||||||
|
* a big dictionary are not a problem in single-call mode. It is enough that
|
||||||
|
* the output buffer is big enough to hold the actual uncompressed data; it
|
||||||
|
* can be smaller than the dictionary size stored in the stream headers.
|
||||||
|
*
|
||||||
|
* Multi-call mode with preallocated dictionary (XZ_PREALLOC): dict_max bytes
|
||||||
|
* of memory is preallocated for the LZMA2 dictionary. This way there is no
|
||||||
|
* risk that xz_dec_run() could run out of memory, since xz_dec_run() will
|
||||||
|
* never allocate any memory. Instead, if the preallocated dictionary is too
|
||||||
|
* small for decoding the given input stream, xz_dec_run() will return
|
||||||
|
* XZ_MEMLIMIT_ERROR. Thus, it is important to know what kind of data will be
|
||||||
|
* decoded to avoid allocating excessive amount of memory for the dictionary.
|
||||||
|
*
|
||||||
|
* Multi-call mode with dynamically allocated dictionary (XZ_DYNALLOC):
|
||||||
|
* dict_max specifies the maximum allowed dictionary size that xz_dec_run()
|
||||||
|
* may allocate once it has parsed the dictionary size from the stream
|
||||||
|
* headers. This way excessive allocations can be avoided while still
|
||||||
|
* limiting the maximum memory usage to a sane value to prevent running the
|
||||||
|
* system out of memory when decompressing streams from untrusted sources.
|
||||||
|
*
|
||||||
|
* On success, xz_dec_init() returns a pointer to struct xz_dec, which is
|
||||||
|
* ready to be used with xz_dec_run(). If memory allocation fails,
|
||||||
|
* xz_dec_init() returns NULL.
|
||||||
|
*/
|
||||||
|
XZ_EXTERN struct xz_dec *xz_dec_init(enum xz_mode mode, uint32_t dict_max);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* xz_dec_run() - Run the XZ decoder for a single XZ stream
|
||||||
|
* @s: Decoder state allocated using xz_dec_init()
|
||||||
|
* @b: Input and output buffers
|
||||||
|
*
|
||||||
|
* The possible return values depend on build options and operation mode.
|
||||||
|
* See enum xz_ret for details.
|
||||||
|
*
|
||||||
|
* Note that if an error occurs in single-call mode (return value is not
|
||||||
|
* XZ_STREAM_END), b->in_pos and b->out_pos are not modified and the
|
||||||
|
* contents of the output buffer from b->out[b->out_pos] onward are
|
||||||
|
* undefined. This is true even after XZ_BUF_ERROR, because with some filter
|
||||||
|
* chains, there may be a second pass over the output buffer, and this pass
|
||||||
|
* cannot be properly done if the output buffer is truncated. Thus, you
|
||||||
|
* cannot give the single-call decoder a too small buffer and then expect to
|
||||||
|
* get that amount valid data from the beginning of the stream. You must use
|
||||||
|
* the multi-call decoder if you don't want to uncompress the whole stream.
|
||||||
|
*
|
||||||
|
* Use xz_dec_run() when XZ data is stored inside some other file format.
|
||||||
|
* The decoding will stop after one XZ stream has been decompressed. To
|
||||||
|
* decompress regular .xz files which might have multiple concatenated
|
||||||
|
* streams, use xz_dec_catrun() instead.
|
||||||
|
*/
|
||||||
|
XZ_EXTERN enum xz_ret xz_dec_run(struct xz_dec *s, struct xz_buf *b);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* xz_dec_catrun() - Run the XZ decoder with support for concatenated streams
|
||||||
|
* @s: Decoder state allocated using xz_dec_init()
|
||||||
|
* @b: Input and output buffers
|
||||||
|
* @finish: This is an int instead of bool to avoid requiring stdbool.h.
|
||||||
|
* As long as more input might be coming, finish must be false.
|
||||||
|
* When the caller knows that it has provided all the input to
|
||||||
|
* the decoder (some possibly still in b->in), it must set finish
|
||||||
|
* to true. Only when finish is true can this function return
|
||||||
|
* XZ_STREAM_END to indicate successful decompression of the
|
||||||
|
* file. In single-call mode (XZ_SINGLE) finish is assumed to
|
||||||
|
* always be true; the caller-provided value is ignored.
|
||||||
|
*
|
||||||
|
* This is like xz_dec_run() except that this makes it easy to decode .xz
|
||||||
|
* files with multiple streams (multiple .xz files concatenated as is).
|
||||||
|
* The rarely-used Stream Padding feature is supported too, that is, there
|
||||||
|
* can be null bytes after or between the streams. The number of null bytes
|
||||||
|
* must be a multiple of four.
|
||||||
|
*
|
||||||
|
* When finish is false and b->in_pos == b->in_size, it is possible that
|
||||||
|
* XZ_BUF_ERROR isn't returned even when no progress is possible (XZ_OK is
|
||||||
|
* returned instead). This shouldn't matter because in this situation a
|
||||||
|
* reasonable caller will attempt to provide more input or set finish to
|
||||||
|
* true for the next xz_dec_catrun() call anyway.
|
||||||
|
*
|
||||||
|
* For any struct xz_dec that has been initialized for multi-call mode:
|
||||||
|
* Once decoding has been started with xz_dec_run() or xz_dec_catrun(),
|
||||||
|
* the same function must be used until xz_dec_reset() or xz_dec_end().
|
||||||
|
* Switching between the two decoding functions without resetting results
|
||||||
|
* in undefined behavior.
|
||||||
|
*
|
||||||
|
* xz_dec_catrun() is only available if XZ_DEC_CONCATENATED was defined
|
||||||
|
* at compile time.
|
||||||
|
*/
|
||||||
|
XZ_EXTERN enum xz_ret xz_dec_catrun(struct xz_dec *s, struct xz_buf *b,
|
||||||
|
int finish);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* xz_dec_reset() - Reset an already allocated decoder state
|
||||||
|
* @s: Decoder state allocated using xz_dec_init()
|
||||||
|
*
|
||||||
|
* This function can be used to reset the multi-call decoder state without
|
||||||
|
* freeing and reallocating memory with xz_dec_end() and xz_dec_init().
|
||||||
|
*
|
||||||
|
* In single-call mode, xz_dec_reset() is always called in the beginning of
|
||||||
|
* xz_dec_run(). Thus, explicit call to xz_dec_reset() is useful only in
|
||||||
|
* multi-call mode.
|
||||||
|
*/
|
||||||
|
XZ_EXTERN void xz_dec_reset(struct xz_dec *s);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* xz_dec_end() - Free the memory allocated for the decoder state
|
||||||
|
* @s: Decoder state allocated using xz_dec_init(). If s is NULL,
|
||||||
|
* this function does nothing.
|
||||||
|
*/
|
||||||
|
XZ_EXTERN void xz_dec_end(struct xz_dec *s);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* DOC: MicroLZMA decompressor
|
||||||
|
*
|
||||||
|
* This MicroLZMA header format was created for use in EROFS but may be used
|
||||||
|
* by others too. **In most cases one needs the XZ APIs above instead.**
|
||||||
|
*
|
||||||
|
* The compressed format supported by this decoder is a raw LZMA stream
|
||||||
|
* whose first byte (always 0x00) has been replaced with bitwise-negation
|
||||||
|
* of the LZMA properties (lc/lp/pb) byte. For example, if lc/lp/pb is
|
||||||
|
* 3/0/2, the first byte is 0xA2. This way the first byte can never be 0x00.
|
||||||
|
* Just like with LZMA2, lc + lp <= 4 must be true. The LZMA end-of-stream
|
||||||
|
* marker must not be used. The unused values are reserved for future use.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* struct xz_dec_microlzma - Opaque type to hold the MicroLZMA decoder state
|
||||||
|
*/
|
||||||
|
struct xz_dec_microlzma;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* xz_dec_microlzma_alloc() - Allocate memory for the MicroLZMA decoder
|
||||||
|
* @mode: XZ_SINGLE or XZ_PREALLOC
|
||||||
|
* @dict_size: LZMA dictionary size. This must be at least 4 KiB and
|
||||||
|
* at most 3 GiB.
|
||||||
|
*
|
||||||
|
* In contrast to xz_dec_init(), this function only allocates the memory
|
||||||
|
* and remembers the dictionary size. xz_dec_microlzma_reset() must be used
|
||||||
|
* before calling xz_dec_microlzma_run().
|
||||||
|
*
|
||||||
|
* The amount of allocated memory is a little less than 30 KiB with XZ_SINGLE.
|
||||||
|
* With XZ_PREALLOC also a dictionary buffer of dict_size bytes is allocated.
|
||||||
|
*
|
||||||
|
* On success, xz_dec_microlzma_alloc() returns a pointer to
|
||||||
|
* struct xz_dec_microlzma. If memory allocation fails or
|
||||||
|
* dict_size is invalid, NULL is returned.
|
||||||
|
*/
|
||||||
|
XZ_EXTERN struct xz_dec_microlzma *xz_dec_microlzma_alloc(enum xz_mode mode,
|
||||||
|
uint32_t dict_size);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* xz_dec_microlzma_reset() - Reset the MicroLZMA decoder state
|
||||||
|
* @s: Decoder state allocated using xz_dec_microlzma_alloc()
|
||||||
|
* @comp_size: Compressed size of the input stream
|
||||||
|
* @uncomp_size: Uncompressed size of the input stream. A value smaller
|
||||||
|
* than the real uncompressed size of the input stream can
|
||||||
|
* be specified if uncomp_size_is_exact is set to false.
|
||||||
|
* uncomp_size can never be set to a value larger than the
|
||||||
|
* expected real uncompressed size because it would eventually
|
||||||
|
* result in XZ_DATA_ERROR.
|
||||||
|
* @uncomp_size_is_exact: This is an int instead of bool to avoid
|
||||||
|
* requiring stdbool.h. This should normally be set to true.
|
||||||
|
* When this is set to false, error detection is weaker.
|
||||||
|
*/
|
||||||
|
XZ_EXTERN void xz_dec_microlzma_reset(struct xz_dec_microlzma *s,
|
||||||
|
uint32_t comp_size, uint32_t uncomp_size,
|
||||||
|
int uncomp_size_is_exact);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* xz_dec_microlzma_run() - Run the MicroLZMA decoder
|
||||||
|
* @s: Decoder state initialized using xz_dec_microlzma_reset()
|
||||||
|
* @b: Input and output buffers
|
||||||
|
*
|
||||||
|
* This works similarly to xz_dec_run() with a few important differences.
|
||||||
|
* Only the differences are documented here.
|
||||||
|
*
|
||||||
|
* The only possible return values are XZ_OK, XZ_STREAM_END, and
|
||||||
|
* XZ_DATA_ERROR. This function cannot return XZ_BUF_ERROR: if no progress
|
||||||
|
* is possible due to lack of input data or output space, this function will
|
||||||
|
* keep returning XZ_OK. Thus, the calling code must be written so that it
|
||||||
|
* will eventually provide input and output space matching (or exceeding)
|
||||||
|
* comp_size and uncomp_size arguments given to xz_dec_microlzma_reset().
|
||||||
|
* If the caller cannot do this (for example, if the input file is truncated
|
||||||
|
* or otherwise corrupt), the caller must detect this error by itself to
|
||||||
|
* avoid an infinite loop.
|
||||||
|
*
|
||||||
|
* If the compressed data seems to be corrupt, XZ_DATA_ERROR is returned.
|
||||||
|
* This can happen also when incorrect dictionary, uncompressed, or
|
||||||
|
* compressed sizes have been specified.
|
||||||
|
*
|
||||||
|
* With XZ_PREALLOC only: As an extra feature, b->out may be NULL to skip over
|
||||||
|
* uncompressed data. This way the caller doesn't need to provide a temporary
|
||||||
|
* output buffer for the bytes that will be ignored.
|
||||||
|
*
|
||||||
|
* With XZ_SINGLE only: In contrast to xz_dec_run(), the return value XZ_OK
|
||||||
|
* is also possible and thus XZ_SINGLE is actually a limited multi-call mode.
|
||||||
|
* After XZ_OK the bytes decoded so far may be read from the output buffer.
|
||||||
|
* It is possible to continue decoding but the variables b->out and b->out_pos
|
||||||
|
* MUST NOT be changed by the caller. Increasing the value of b->out_size is
|
||||||
|
* allowed to make more output space available; one doesn't need to provide
|
||||||
|
* space for the whole uncompressed data on the first call. The input buffer
|
||||||
|
* may be changed normally like with XZ_PREALLOC. This way input data can be
|
||||||
|
* provided from non-contiguous memory.
|
||||||
|
*/
|
||||||
|
XZ_EXTERN enum xz_ret xz_dec_microlzma_run(struct xz_dec_microlzma *s,
|
||||||
|
struct xz_buf *b);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* xz_dec_microlzma_end() - Free the memory allocated for the decoder state
|
||||||
|
* @s: Decoder state allocated using xz_dec_microlzma_alloc().
|
||||||
|
* If s is NULL, this function does nothing.
|
||||||
|
*/
|
||||||
|
XZ_EXTERN void xz_dec_microlzma_end(struct xz_dec_microlzma *s);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Standalone build (userspace build or in-kernel build for boot time use)
|
||||||
|
* needs a CRC32 implementation. For normal in-kernel use, kernel's own
|
||||||
|
* CRC32 module is used instead, and users of this module don't need to
|
||||||
|
* care about the functions below.
|
||||||
|
*/
|
||||||
|
#ifndef XZ_INTERNAL_CRC32
|
||||||
|
# ifdef __KERNEL__
|
||||||
|
# define XZ_INTERNAL_CRC32 0
|
||||||
|
# else
|
||||||
|
# define XZ_INTERNAL_CRC32 1
|
||||||
|
# endif
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If CRC64 support has been enabled with XZ_USE_CRC64, a CRC64
|
||||||
|
* implementation is needed too.
|
||||||
|
*/
|
||||||
|
#ifndef XZ_USE_CRC64
|
||||||
|
# undef XZ_INTERNAL_CRC64
|
||||||
|
# define XZ_INTERNAL_CRC64 0
|
||||||
|
#endif
|
||||||
|
#ifndef XZ_INTERNAL_CRC64
|
||||||
|
# ifdef __KERNEL__
|
||||||
|
# error Using CRC64 in the kernel has not been implemented.
|
||||||
|
# else
|
||||||
|
# define XZ_INTERNAL_CRC64 1
|
||||||
|
# endif
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if XZ_INTERNAL_CRC32
|
||||||
|
/*
|
||||||
|
* This must be called before any other xz_* function to initialize
|
||||||
|
* the CRC32 lookup table.
|
||||||
|
*/
|
||||||
|
XZ_EXTERN void xz_crc32_init(void);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Update CRC32 value using the polynomial from IEEE-802.3. To start a new
|
||||||
|
* calculation, the third argument must be zero. To continue the calculation,
|
||||||
|
* the previously returned value is passed as the third argument.
|
||||||
|
*/
|
||||||
|
XZ_EXTERN uint32_t xz_crc32(const uint8_t *buf, size_t size, uint32_t crc);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if XZ_INTERNAL_CRC64
|
||||||
|
/*
|
||||||
|
* This must be called before any other xz_* function (except xz_crc32_init())
|
||||||
|
* to initialize the CRC64 lookup table.
|
||||||
|
*/
|
||||||
|
XZ_EXTERN void xz_crc64_init(void);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Update CRC64 value using the polynomial from ECMA-182. To start a new
|
||||||
|
* calculation, the third argument must be zero. To continue the calculation,
|
||||||
|
* the previously returned value is passed as the third argument.
|
||||||
|
*/
|
||||||
|
XZ_EXTERN uint64_t xz_crc64(const uint8_t *buf, size_t size, uint64_t crc);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
||||||
138
android/app/src/main/cpp/xz/xz_config.h
Normal file
138
android/app/src/main/cpp/xz/xz_config.h
Normal file
@@ -0,0 +1,138 @@
|
|||||||
|
/* SPDX-License-Identifier: 0BSD */
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Private includes and definitions for userspace use of XZ Embedded
|
||||||
|
*
|
||||||
|
* Author: Lasse Collin <lasse.collin@tukaani.org>
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef XZ_CONFIG_H
|
||||||
|
#define XZ_CONFIG_H
|
||||||
|
|
||||||
|
/* Uncomment to enable building of xz_dec_catrun(). */
|
||||||
|
/* #define XZ_DEC_CONCATENATED */
|
||||||
|
|
||||||
|
/* Uncomment to enable CRC64 support. */
|
||||||
|
/* #define XZ_USE_CRC64 */
|
||||||
|
|
||||||
|
/* Uncomment as needed to enable BCJ filter decoders. */
|
||||||
|
/* #define XZ_DEC_X86 */
|
||||||
|
/* #define XZ_DEC_ARM */
|
||||||
|
/* #define XZ_DEC_ARMTHUMB */
|
||||||
|
/* #define XZ_DEC_ARM64 */
|
||||||
|
/* #define XZ_DEC_RISCV */
|
||||||
|
/* #define XZ_DEC_POWERPC */
|
||||||
|
/* #define XZ_DEC_IA64 */
|
||||||
|
/* #define XZ_DEC_SPARC */
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Visual Studio 2013 update 2 supports only __inline, not inline.
|
||||||
|
* MSVC v19.0 / VS 2015 and newer support both.
|
||||||
|
*/
|
||||||
|
#if defined(_MSC_VER) && _MSC_VER < 1900 && !defined(inline)
|
||||||
|
# define inline __inline
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include <stdbool.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
#include "xz.h"
|
||||||
|
|
||||||
|
#define kmalloc(size, flags) malloc(size)
|
||||||
|
#define kfree(ptr) free(ptr)
|
||||||
|
#define vmalloc(size) malloc(size)
|
||||||
|
#define vfree(ptr) free(ptr)
|
||||||
|
|
||||||
|
#define memeq(a, b, size) (memcmp(a, b, size) == 0)
|
||||||
|
#define memzero(buf, size) memset(buf, 0, size)
|
||||||
|
|
||||||
|
#ifndef min
|
||||||
|
# define min(x, y) ((x) < (y) ? (x) : (y))
|
||||||
|
#endif
|
||||||
|
#define min_t(type, x, y) min(x, y)
|
||||||
|
|
||||||
|
#ifndef fallthrough
|
||||||
|
# if defined(__STDC_VERSION__) && __STDC_VERSION__ >= 202311
|
||||||
|
# define fallthrough [[fallthrough]]
|
||||||
|
# elif (defined(__GNUC__) && __GNUC__ >= 7) \
|
||||||
|
|| (defined(__clang_major__) && __clang_major__ >= 10)
|
||||||
|
# define fallthrough __attribute__((__fallthrough__))
|
||||||
|
# else
|
||||||
|
# define fallthrough do {} while (0)
|
||||||
|
# endif
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Some functions have been marked with __always_inline to keep the
|
||||||
|
* performance reasonable even when the compiler is optimizing for
|
||||||
|
* small code size. You may be able to save a few bytes by #defining
|
||||||
|
* __always_inline to plain inline, but don't complain if the code
|
||||||
|
* becomes slow.
|
||||||
|
*
|
||||||
|
* NOTE: System headers on GNU/Linux may #define this macro already,
|
||||||
|
* so if you want to change it, you need to #undef it first.
|
||||||
|
*/
|
||||||
|
#ifndef __always_inline
|
||||||
|
# ifdef __GNUC__
|
||||||
|
# define __always_inline \
|
||||||
|
inline __attribute__((__always_inline__))
|
||||||
|
# else
|
||||||
|
# define __always_inline inline
|
||||||
|
# endif
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* Inline functions to access unaligned unsigned 32-bit integers */
|
||||||
|
#ifndef get_unaligned_le32
|
||||||
|
static inline uint32_t get_unaligned_le32(const uint8_t *buf)
|
||||||
|
{
|
||||||
|
return (uint32_t)buf[0]
|
||||||
|
| ((uint32_t)buf[1] << 8)
|
||||||
|
| ((uint32_t)buf[2] << 16)
|
||||||
|
| ((uint32_t)buf[3] << 24);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifndef get_unaligned_be32
|
||||||
|
static inline uint32_t get_unaligned_be32(const uint8_t *buf)
|
||||||
|
{
|
||||||
|
return (uint32_t)((uint32_t)buf[0] << 24)
|
||||||
|
| ((uint32_t)buf[1] << 16)
|
||||||
|
| ((uint32_t)buf[2] << 8)
|
||||||
|
| (uint32_t)buf[3];
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifndef put_unaligned_le32
|
||||||
|
static inline void put_unaligned_le32(uint32_t val, uint8_t *buf)
|
||||||
|
{
|
||||||
|
buf[0] = (uint8_t)val;
|
||||||
|
buf[1] = (uint8_t)(val >> 8);
|
||||||
|
buf[2] = (uint8_t)(val >> 16);
|
||||||
|
buf[3] = (uint8_t)(val >> 24);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifndef put_unaligned_be32
|
||||||
|
static inline void put_unaligned_be32(uint32_t val, uint8_t *buf)
|
||||||
|
{
|
||||||
|
buf[0] = (uint8_t)(val >> 24);
|
||||||
|
buf[1] = (uint8_t)(val >> 16);
|
||||||
|
buf[2] = (uint8_t)(val >> 8);
|
||||||
|
buf[3] = (uint8_t)val;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/*
|
||||||
|
* To keep things simpler, use the generic unaligned methods also for
|
||||||
|
* aligned access. The only place where performance could matter is
|
||||||
|
* SHA-256 but files using SHA-256 aren't common.
|
||||||
|
*/
|
||||||
|
#ifndef get_le32
|
||||||
|
# define get_le32 get_unaligned_le32
|
||||||
|
#endif
|
||||||
|
#ifndef get_be32
|
||||||
|
# define get_be32 get_unaligned_be32
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
||||||
58
android/app/src/main/cpp/xz/xz_crc32.c
Normal file
58
android/app/src/main/cpp/xz/xz_crc32.c
Normal file
@@ -0,0 +1,58 @@
|
|||||||
|
// SPDX-License-Identifier: 0BSD
|
||||||
|
|
||||||
|
/*
|
||||||
|
* CRC32 using the polynomial from IEEE-802.3
|
||||||
|
*
|
||||||
|
* Authors: Lasse Collin <lasse.collin@tukaani.org>
|
||||||
|
* Igor Pavlov <https://7-zip.org/>
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This is not the fastest implementation, but it is pretty compact.
|
||||||
|
* The fastest versions of xz_crc32() on modern CPUs without hardware
|
||||||
|
* accelerated CRC instruction are 3-5 times as fast as this version,
|
||||||
|
* but they are bigger and use more memory for the lookup table.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "xz_private.h"
|
||||||
|
|
||||||
|
/*
|
||||||
|
* STATIC_RW_DATA is used in the pre-boot environment on some architectures.
|
||||||
|
* See <linux/decompress/mm.h> for details.
|
||||||
|
*/
|
||||||
|
#ifndef STATIC_RW_DATA
|
||||||
|
# define STATIC_RW_DATA static
|
||||||
|
#endif
|
||||||
|
|
||||||
|
STATIC_RW_DATA uint32_t xz_crc32_table[256];
|
||||||
|
|
||||||
|
XZ_EXTERN void xz_crc32_init(void)
|
||||||
|
{
|
||||||
|
const uint32_t poly = 0xEDB88320;
|
||||||
|
|
||||||
|
uint32_t i;
|
||||||
|
uint32_t j;
|
||||||
|
uint32_t r;
|
||||||
|
|
||||||
|
for (i = 0; i < 256; ++i) {
|
||||||
|
r = i;
|
||||||
|
for (j = 0; j < 8; ++j)
|
||||||
|
r = (r >> 1) ^ (poly & ~((r & 1) - 1));
|
||||||
|
|
||||||
|
xz_crc32_table[i] = r;
|
||||||
|
}
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
XZ_EXTERN uint32_t xz_crc32(const uint8_t *buf, size_t size, uint32_t crc)
|
||||||
|
{
|
||||||
|
crc = ~crc;
|
||||||
|
|
||||||
|
while (size != 0) {
|
||||||
|
crc = xz_crc32_table[*buf++ ^ (crc & 0xFF)] ^ (crc >> 8);
|
||||||
|
--size;
|
||||||
|
}
|
||||||
|
|
||||||
|
return ~crc;
|
||||||
|
}
|
||||||
53
android/app/src/main/cpp/xz/xz_crc64.c
Normal file
53
android/app/src/main/cpp/xz/xz_crc64.c
Normal file
@@ -0,0 +1,53 @@
|
|||||||
|
// SPDX-License-Identifier: 0BSD
|
||||||
|
|
||||||
|
/*
|
||||||
|
* CRC64 using the polynomial from ECMA-182
|
||||||
|
*
|
||||||
|
* This file is similar to xz_crc32.c. See the comments there.
|
||||||
|
*
|
||||||
|
* Authors: Lasse Collin <lasse.collin@tukaani.org>
|
||||||
|
* Igor Pavlov <https://7-zip.org/>
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "xz_private.h"
|
||||||
|
|
||||||
|
#ifndef STATIC_RW_DATA
|
||||||
|
# define STATIC_RW_DATA static
|
||||||
|
#endif
|
||||||
|
|
||||||
|
STATIC_RW_DATA uint64_t xz_crc64_table[256];
|
||||||
|
|
||||||
|
XZ_EXTERN void xz_crc64_init(void)
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
* The ULL suffix is needed for -std=gnu89 compatibility
|
||||||
|
* on 32-bit platforms.
|
||||||
|
*/
|
||||||
|
const uint64_t poly = 0xC96C5795D7870F42ULL;
|
||||||
|
|
||||||
|
uint32_t i;
|
||||||
|
uint32_t j;
|
||||||
|
uint64_t r;
|
||||||
|
|
||||||
|
for (i = 0; i < 256; ++i) {
|
||||||
|
r = i;
|
||||||
|
for (j = 0; j < 8; ++j)
|
||||||
|
r = (r >> 1) ^ (poly & ~((r & 1) - 1));
|
||||||
|
|
||||||
|
xz_crc64_table[i] = r;
|
||||||
|
}
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
XZ_EXTERN uint64_t xz_crc64(const uint8_t *buf, size_t size, uint64_t crc)
|
||||||
|
{
|
||||||
|
crc = ~crc;
|
||||||
|
|
||||||
|
while (size != 0) {
|
||||||
|
crc = xz_crc64_table[*buf++ ^ (crc & 0xFF)] ^ (crc >> 8);
|
||||||
|
--size;
|
||||||
|
}
|
||||||
|
|
||||||
|
return ~crc;
|
||||||
|
}
|
||||||
738
android/app/src/main/cpp/xz/xz_dec_bcj.c
Normal file
738
android/app/src/main/cpp/xz/xz_dec_bcj.c
Normal file
@@ -0,0 +1,738 @@
|
|||||||
|
// SPDX-License-Identifier: 0BSD
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Branch/Call/Jump (BCJ) filter decoders
|
||||||
|
*
|
||||||
|
* Authors: Lasse Collin <lasse.collin@tukaani.org>
|
||||||
|
* Igor Pavlov <https://7-zip.org/>
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "xz_private.h"
|
||||||
|
|
||||||
|
/*
|
||||||
|
* The rest of the file is inside this ifdef. It makes things a little more
|
||||||
|
* convenient when building without support for any BCJ filters.
|
||||||
|
*/
|
||||||
|
#ifdef XZ_DEC_BCJ
|
||||||
|
|
||||||
|
struct xz_dec_bcj {
|
||||||
|
/* Type of the BCJ filter being used */
|
||||||
|
enum {
|
||||||
|
BCJ_X86 = 4, /* x86 or x86-64 */
|
||||||
|
BCJ_POWERPC = 5, /* Big endian only */
|
||||||
|
BCJ_IA64 = 6, /* Big or little endian */
|
||||||
|
BCJ_ARM = 7, /* Little endian only */
|
||||||
|
BCJ_ARMTHUMB = 8, /* Little endian only */
|
||||||
|
BCJ_SPARC = 9, /* Big or little endian */
|
||||||
|
BCJ_ARM64 = 10, /* AArch64 */
|
||||||
|
BCJ_RISCV = 11 /* RV32GQC_Zfh, RV64GQC_Zfh */
|
||||||
|
} type;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Return value of the next filter in the chain. We need to preserve
|
||||||
|
* this information across calls, because we must not call the next
|
||||||
|
* filter anymore once it has returned XZ_STREAM_END.
|
||||||
|
*/
|
||||||
|
enum xz_ret ret;
|
||||||
|
|
||||||
|
/* True if we are operating in single-call mode. */
|
||||||
|
bool single_call;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Absolute position relative to the beginning of the uncompressed
|
||||||
|
* data (in a single .xz Block). We care only about the lowest 32
|
||||||
|
* bits so this doesn't need to be uint64_t even with big files.
|
||||||
|
*/
|
||||||
|
uint32_t pos;
|
||||||
|
|
||||||
|
/* x86 filter state */
|
||||||
|
uint32_t x86_prev_mask;
|
||||||
|
|
||||||
|
/* Temporary space to hold the variables from struct xz_buf */
|
||||||
|
uint8_t *out;
|
||||||
|
size_t out_pos;
|
||||||
|
size_t out_size;
|
||||||
|
|
||||||
|
struct {
|
||||||
|
/* Amount of already filtered data in the beginning of buf */
|
||||||
|
size_t filtered;
|
||||||
|
|
||||||
|
/* Total amount of data currently stored in buf */
|
||||||
|
size_t size;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Buffer to hold a mix of filtered and unfiltered data. This
|
||||||
|
* needs to be big enough to hold Alignment + 2 * Look-ahead:
|
||||||
|
*
|
||||||
|
* Type Alignment Look-ahead
|
||||||
|
* x86 1 4
|
||||||
|
* PowerPC 4 0
|
||||||
|
* IA-64 16 0
|
||||||
|
* ARM 4 0
|
||||||
|
* ARM-Thumb 2 2
|
||||||
|
* SPARC 4 0
|
||||||
|
*/
|
||||||
|
uint8_t buf[16];
|
||||||
|
} temp;
|
||||||
|
};
|
||||||
|
|
||||||
|
#ifdef XZ_DEC_X86
|
||||||
|
/*
|
||||||
|
* This is used to test the most significant byte of a memory address
|
||||||
|
* in an x86 instruction.
|
||||||
|
*/
|
||||||
|
static inline int bcj_x86_test_msbyte(uint8_t b)
|
||||||
|
{
|
||||||
|
return b == 0x00 || b == 0xFF;
|
||||||
|
}
|
||||||
|
|
||||||
|
static size_t bcj_x86(struct xz_dec_bcj *s, uint8_t *buf, size_t size)
|
||||||
|
{
|
||||||
|
static const bool mask_to_allowed_status[8]
|
||||||
|
= { true, true, true, false, true, false, false, false };
|
||||||
|
|
||||||
|
static const uint8_t mask_to_bit_num[8] = { 0, 1, 2, 2, 3, 3, 3, 3 };
|
||||||
|
|
||||||
|
size_t i;
|
||||||
|
size_t prev_pos = (size_t)-1;
|
||||||
|
uint32_t prev_mask = s->x86_prev_mask;
|
||||||
|
uint32_t src;
|
||||||
|
uint32_t dest;
|
||||||
|
uint32_t j;
|
||||||
|
uint8_t b;
|
||||||
|
|
||||||
|
if (size <= 4)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
size -= 4;
|
||||||
|
for (i = 0; i < size; ++i) {
|
||||||
|
if ((buf[i] & 0xFE) != 0xE8)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
prev_pos = i - prev_pos;
|
||||||
|
if (prev_pos > 3) {
|
||||||
|
prev_mask = 0;
|
||||||
|
} else {
|
||||||
|
prev_mask = (prev_mask << (prev_pos - 1)) & 7;
|
||||||
|
if (prev_mask != 0) {
|
||||||
|
b = buf[i + 4 - mask_to_bit_num[prev_mask]];
|
||||||
|
if (!mask_to_allowed_status[prev_mask]
|
||||||
|
|| bcj_x86_test_msbyte(b)) {
|
||||||
|
prev_pos = i;
|
||||||
|
prev_mask = (prev_mask << 1) | 1;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
prev_pos = i;
|
||||||
|
|
||||||
|
if (bcj_x86_test_msbyte(buf[i + 4])) {
|
||||||
|
src = get_unaligned_le32(buf + i + 1);
|
||||||
|
while (true) {
|
||||||
|
dest = src - (s->pos + (uint32_t)i + 5);
|
||||||
|
if (prev_mask == 0)
|
||||||
|
break;
|
||||||
|
|
||||||
|
j = mask_to_bit_num[prev_mask] * 8;
|
||||||
|
b = (uint8_t)(dest >> (24 - j));
|
||||||
|
if (!bcj_x86_test_msbyte(b))
|
||||||
|
break;
|
||||||
|
|
||||||
|
src = dest ^ (((uint32_t)1 << (32 - j)) - 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
dest &= 0x01FFFFFF;
|
||||||
|
dest |= (uint32_t)0 - (dest & 0x01000000);
|
||||||
|
put_unaligned_le32(dest, buf + i + 1);
|
||||||
|
i += 4;
|
||||||
|
} else {
|
||||||
|
prev_mask = (prev_mask << 1) | 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
prev_pos = i - prev_pos;
|
||||||
|
s->x86_prev_mask = prev_pos > 3 ? 0 : prev_mask << (prev_pos - 1);
|
||||||
|
return i;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef XZ_DEC_POWERPC
|
||||||
|
static size_t bcj_powerpc(struct xz_dec_bcj *s, uint8_t *buf, size_t size)
|
||||||
|
{
|
||||||
|
size_t i;
|
||||||
|
uint32_t instr;
|
||||||
|
|
||||||
|
size &= ~(size_t)3;
|
||||||
|
|
||||||
|
for (i = 0; i < size; i += 4) {
|
||||||
|
instr = get_unaligned_be32(buf + i);
|
||||||
|
if ((instr & 0xFC000003) == 0x48000001) {
|
||||||
|
instr &= 0x03FFFFFC;
|
||||||
|
instr -= s->pos + (uint32_t)i;
|
||||||
|
instr &= 0x03FFFFFC;
|
||||||
|
instr |= 0x48000001;
|
||||||
|
put_unaligned_be32(instr, buf + i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return i;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef XZ_DEC_IA64
|
||||||
|
static size_t bcj_ia64(struct xz_dec_bcj *s, uint8_t *buf, size_t size)
|
||||||
|
{
|
||||||
|
static const uint8_t branch_table[32] = {
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
0, 0, 0, 0, 0, 0, 0, 0,
|
||||||
|
4, 4, 6, 6, 0, 0, 7, 7,
|
||||||
|
4, 4, 0, 0, 4, 4, 0, 0
|
||||||
|
};
|
||||||
|
|
||||||
|
/*
|
||||||
|
* The local variables take a little bit stack space, but it's less
|
||||||
|
* than what LZMA2 decoder takes, so it doesn't make sense to reduce
|
||||||
|
* stack usage here without doing that for the LZMA2 decoder too.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* Loop counters */
|
||||||
|
size_t i;
|
||||||
|
size_t j;
|
||||||
|
|
||||||
|
/* Instruction slot (0, 1, or 2) in the 128-bit instruction word */
|
||||||
|
uint32_t slot;
|
||||||
|
|
||||||
|
/* Bitwise offset of the instruction indicated by slot */
|
||||||
|
uint32_t bit_pos;
|
||||||
|
|
||||||
|
/* bit_pos split into byte and bit parts */
|
||||||
|
uint32_t byte_pos;
|
||||||
|
uint32_t bit_res;
|
||||||
|
|
||||||
|
/* Address part of an instruction */
|
||||||
|
uint32_t addr;
|
||||||
|
|
||||||
|
/* Mask used to detect which instructions to convert */
|
||||||
|
uint32_t mask;
|
||||||
|
|
||||||
|
/* 41-bit instruction stored somewhere in the lowest 48 bits */
|
||||||
|
uint64_t instr;
|
||||||
|
|
||||||
|
/* Instruction normalized with bit_res for easier manipulation */
|
||||||
|
uint64_t norm;
|
||||||
|
|
||||||
|
size &= ~(size_t)15;
|
||||||
|
|
||||||
|
for (i = 0; i < size; i += 16) {
|
||||||
|
mask = branch_table[buf[i] & 0x1F];
|
||||||
|
for (slot = 0, bit_pos = 5; slot < 3; ++slot, bit_pos += 41) {
|
||||||
|
if (((mask >> slot) & 1) == 0)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
byte_pos = bit_pos >> 3;
|
||||||
|
bit_res = bit_pos & 7;
|
||||||
|
instr = 0;
|
||||||
|
for (j = 0; j < 6; ++j)
|
||||||
|
instr |= (uint64_t)(buf[i + j + byte_pos])
|
||||||
|
<< (8 * j);
|
||||||
|
|
||||||
|
norm = instr >> bit_res;
|
||||||
|
|
||||||
|
if (((norm >> 37) & 0x0F) == 0x05
|
||||||
|
&& ((norm >> 9) & 0x07) == 0) {
|
||||||
|
addr = (norm >> 13) & 0x0FFFFF;
|
||||||
|
addr |= ((uint32_t)(norm >> 36) & 1) << 20;
|
||||||
|
addr <<= 4;
|
||||||
|
addr -= s->pos + (uint32_t)i;
|
||||||
|
addr >>= 4;
|
||||||
|
|
||||||
|
norm &= ~((uint64_t)0x8FFFFF << 13);
|
||||||
|
norm |= (uint64_t)(addr & 0x0FFFFF) << 13;
|
||||||
|
norm |= (uint64_t)(addr & 0x100000)
|
||||||
|
<< (36 - 20);
|
||||||
|
|
||||||
|
instr &= (1 << bit_res) - 1;
|
||||||
|
instr |= norm << bit_res;
|
||||||
|
|
||||||
|
for (j = 0; j < 6; j++)
|
||||||
|
buf[i + j + byte_pos]
|
||||||
|
= (uint8_t)(instr >> (8 * j));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return i;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef XZ_DEC_ARM
|
||||||
|
static size_t bcj_arm(struct xz_dec_bcj *s, uint8_t *buf, size_t size)
|
||||||
|
{
|
||||||
|
size_t i;
|
||||||
|
uint32_t addr;
|
||||||
|
|
||||||
|
size &= ~(size_t)3;
|
||||||
|
|
||||||
|
for (i = 0; i < size; i += 4) {
|
||||||
|
if (buf[i + 3] == 0xEB) {
|
||||||
|
addr = (uint32_t)buf[i] | ((uint32_t)buf[i + 1] << 8)
|
||||||
|
| ((uint32_t)buf[i + 2] << 16);
|
||||||
|
addr <<= 2;
|
||||||
|
addr -= s->pos + (uint32_t)i + 8;
|
||||||
|
addr >>= 2;
|
||||||
|
buf[i] = (uint8_t)addr;
|
||||||
|
buf[i + 1] = (uint8_t)(addr >> 8);
|
||||||
|
buf[i + 2] = (uint8_t)(addr >> 16);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return i;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef XZ_DEC_ARMTHUMB
|
||||||
|
static size_t bcj_armthumb(struct xz_dec_bcj *s, uint8_t *buf, size_t size)
|
||||||
|
{
|
||||||
|
size_t i;
|
||||||
|
uint32_t addr;
|
||||||
|
|
||||||
|
if (size < 4)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
size -= 4;
|
||||||
|
|
||||||
|
for (i = 0; i <= size; i += 2) {
|
||||||
|
if ((buf[i + 1] & 0xF8) == 0xF0
|
||||||
|
&& (buf[i + 3] & 0xF8) == 0xF8) {
|
||||||
|
addr = (((uint32_t)buf[i + 1] & 0x07) << 19)
|
||||||
|
| ((uint32_t)buf[i] << 11)
|
||||||
|
| (((uint32_t)buf[i + 3] & 0x07) << 8)
|
||||||
|
| (uint32_t)buf[i + 2];
|
||||||
|
addr <<= 1;
|
||||||
|
addr -= s->pos + (uint32_t)i + 4;
|
||||||
|
addr >>= 1;
|
||||||
|
buf[i + 1] = (uint8_t)(0xF0 | ((addr >> 19) & 0x07));
|
||||||
|
buf[i] = (uint8_t)(addr >> 11);
|
||||||
|
buf[i + 3] = (uint8_t)(0xF8 | ((addr >> 8) & 0x07));
|
||||||
|
buf[i + 2] = (uint8_t)addr;
|
||||||
|
i += 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return i;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef XZ_DEC_SPARC
|
||||||
|
static size_t bcj_sparc(struct xz_dec_bcj *s, uint8_t *buf, size_t size)
|
||||||
|
{
|
||||||
|
size_t i;
|
||||||
|
uint32_t instr;
|
||||||
|
|
||||||
|
size &= ~(size_t)3;
|
||||||
|
|
||||||
|
for (i = 0; i < size; i += 4) {
|
||||||
|
instr = get_unaligned_be32(buf + i);
|
||||||
|
if ((instr >> 22) == 0x100 || (instr >> 22) == 0x1FF) {
|
||||||
|
instr <<= 2;
|
||||||
|
instr -= s->pos + (uint32_t)i;
|
||||||
|
instr >>= 2;
|
||||||
|
instr = ((uint32_t)0x40000000 - (instr & 0x400000))
|
||||||
|
| 0x40000000 | (instr & 0x3FFFFF);
|
||||||
|
put_unaligned_be32(instr, buf + i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return i;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef XZ_DEC_ARM64
|
||||||
|
static size_t bcj_arm64(struct xz_dec_bcj *s, uint8_t *buf, size_t size)
|
||||||
|
{
|
||||||
|
size_t i;
|
||||||
|
uint32_t instr;
|
||||||
|
uint32_t addr;
|
||||||
|
|
||||||
|
size &= ~(size_t)3;
|
||||||
|
|
||||||
|
for (i = 0; i < size; i += 4) {
|
||||||
|
instr = get_unaligned_le32(buf + i);
|
||||||
|
|
||||||
|
if ((instr >> 26) == 0x25) {
|
||||||
|
/* BL instruction */
|
||||||
|
addr = instr - ((s->pos + (uint32_t)i) >> 2);
|
||||||
|
instr = 0x94000000 | (addr & 0x03FFFFFF);
|
||||||
|
put_unaligned_le32(instr, buf + i);
|
||||||
|
|
||||||
|
} else if ((instr & 0x9F000000) == 0x90000000) {
|
||||||
|
/* ADRP instruction */
|
||||||
|
addr = ((instr >> 29) & 3) | ((instr >> 3) & 0x1FFFFC);
|
||||||
|
|
||||||
|
/* Only convert values in the range +/-512 MiB. */
|
||||||
|
if ((addr + 0x020000) & 0x1C0000)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
addr -= (s->pos + (uint32_t)i) >> 12;
|
||||||
|
|
||||||
|
instr &= 0x9000001F;
|
||||||
|
instr |= (addr & 3) << 29;
|
||||||
|
instr |= (addr & 0x03FFFC) << 3;
|
||||||
|
instr |= (0U - (addr & 0x020000)) & 0xE00000;
|
||||||
|
|
||||||
|
put_unaligned_le32(instr, buf + i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return i;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef XZ_DEC_RISCV
|
||||||
|
static size_t bcj_riscv(struct xz_dec_bcj *s, uint8_t *buf, size_t size)
|
||||||
|
{
|
||||||
|
size_t i;
|
||||||
|
uint32_t b1;
|
||||||
|
uint32_t b2;
|
||||||
|
uint32_t b3;
|
||||||
|
uint32_t instr;
|
||||||
|
uint32_t instr2;
|
||||||
|
uint32_t instr2_rs1;
|
||||||
|
uint32_t addr;
|
||||||
|
|
||||||
|
if (size < 8)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
size -= 8;
|
||||||
|
|
||||||
|
for (i = 0; i <= size; i += 2) {
|
||||||
|
instr = buf[i];
|
||||||
|
|
||||||
|
if (instr == 0xEF) {
|
||||||
|
/* JAL */
|
||||||
|
b1 = buf[i + 1];
|
||||||
|
if ((b1 & 0x0D) != 0)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
b2 = buf[i + 2];
|
||||||
|
b3 = buf[i + 3];
|
||||||
|
|
||||||
|
addr = ((b1 & 0xF0) << 13) | (b2 << 9) | (b3 << 1);
|
||||||
|
addr -= s->pos + (uint32_t)i;
|
||||||
|
|
||||||
|
buf[i + 1] = (uint8_t)((b1 & 0x0F)
|
||||||
|
| ((addr >> 8) & 0xF0));
|
||||||
|
|
||||||
|
buf[i + 2] = (uint8_t)(((addr >> 16) & 0x0F)
|
||||||
|
| ((addr >> 7) & 0x10)
|
||||||
|
| ((addr << 4) & 0xE0));
|
||||||
|
|
||||||
|
buf[i + 3] = (uint8_t)(((addr >> 4) & 0x7F)
|
||||||
|
| ((addr >> 13) & 0x80));
|
||||||
|
|
||||||
|
i += 4 - 2;
|
||||||
|
|
||||||
|
} else if ((instr & 0x7F) == 0x17) {
|
||||||
|
/* AUIPC */
|
||||||
|
instr |= (uint32_t)buf[i + 1] << 8;
|
||||||
|
instr |= (uint32_t)buf[i + 2] << 16;
|
||||||
|
instr |= (uint32_t)buf[i + 3] << 24;
|
||||||
|
|
||||||
|
if (instr & 0xE80) {
|
||||||
|
/* AUIPC's rd doesn't equal x0 or x2. */
|
||||||
|
instr2 = get_unaligned_le32(buf + i + 4);
|
||||||
|
|
||||||
|
if (((instr << 8) ^ (instr2 - 3)) & 0xF8003) {
|
||||||
|
i += 6 - 2;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
addr = (instr & 0xFFFFF000) + (instr2 >> 20);
|
||||||
|
|
||||||
|
instr = 0x17 | (2 << 7) | (instr2 << 12);
|
||||||
|
instr2 = addr;
|
||||||
|
} else {
|
||||||
|
/* AUIPC's rd equals x0 or x2. */
|
||||||
|
instr2_rs1 = instr >> 27;
|
||||||
|
|
||||||
|
if ((uint32_t)((instr - 0x3117) << 18)
|
||||||
|
>= (instr2_rs1 & 0x1D)) {
|
||||||
|
i += 4 - 2;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
addr = get_unaligned_be32(buf + i + 4);
|
||||||
|
addr -= s->pos + (uint32_t)i;
|
||||||
|
|
||||||
|
instr2 = (instr >> 12) | (addr << 20);
|
||||||
|
|
||||||
|
instr = 0x17 | (instr2_rs1 << 7)
|
||||||
|
| ((addr + 0x800) & 0xFFFFF000);
|
||||||
|
}
|
||||||
|
|
||||||
|
put_unaligned_le32(instr, buf + i);
|
||||||
|
put_unaligned_le32(instr2, buf + i + 4);
|
||||||
|
|
||||||
|
i += 8 - 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return i;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Apply the selected BCJ filter. Update *pos and s->pos to match the amount
|
||||||
|
* of data that got filtered.
|
||||||
|
*
|
||||||
|
* NOTE: This is implemented as a switch statement to avoid using function
|
||||||
|
* pointers, which could be problematic in the kernel boot code, which must
|
||||||
|
* avoid pointers to static data (at least on x86).
|
||||||
|
*/
|
||||||
|
static void bcj_apply(struct xz_dec_bcj *s,
|
||||||
|
uint8_t *buf, size_t *pos, size_t size)
|
||||||
|
{
|
||||||
|
size_t filtered;
|
||||||
|
|
||||||
|
buf += *pos;
|
||||||
|
size -= *pos;
|
||||||
|
|
||||||
|
switch (s->type) {
|
||||||
|
#ifdef XZ_DEC_X86
|
||||||
|
case BCJ_X86:
|
||||||
|
filtered = bcj_x86(s, buf, size);
|
||||||
|
break;
|
||||||
|
#endif
|
||||||
|
#ifdef XZ_DEC_POWERPC
|
||||||
|
case BCJ_POWERPC:
|
||||||
|
filtered = bcj_powerpc(s, buf, size);
|
||||||
|
break;
|
||||||
|
#endif
|
||||||
|
#ifdef XZ_DEC_IA64
|
||||||
|
case BCJ_IA64:
|
||||||
|
filtered = bcj_ia64(s, buf, size);
|
||||||
|
break;
|
||||||
|
#endif
|
||||||
|
#ifdef XZ_DEC_ARM
|
||||||
|
case BCJ_ARM:
|
||||||
|
filtered = bcj_arm(s, buf, size);
|
||||||
|
break;
|
||||||
|
#endif
|
||||||
|
#ifdef XZ_DEC_ARMTHUMB
|
||||||
|
case BCJ_ARMTHUMB:
|
||||||
|
filtered = bcj_armthumb(s, buf, size);
|
||||||
|
break;
|
||||||
|
#endif
|
||||||
|
#ifdef XZ_DEC_SPARC
|
||||||
|
case BCJ_SPARC:
|
||||||
|
filtered = bcj_sparc(s, buf, size);
|
||||||
|
break;
|
||||||
|
#endif
|
||||||
|
#ifdef XZ_DEC_ARM64
|
||||||
|
case BCJ_ARM64:
|
||||||
|
filtered = bcj_arm64(s, buf, size);
|
||||||
|
break;
|
||||||
|
#endif
|
||||||
|
#ifdef XZ_DEC_RISCV
|
||||||
|
case BCJ_RISCV:
|
||||||
|
filtered = bcj_riscv(s, buf, size);
|
||||||
|
break;
|
||||||
|
#endif
|
||||||
|
default:
|
||||||
|
/* Never reached but silence compiler warnings. */
|
||||||
|
filtered = 0;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
*pos += filtered;
|
||||||
|
s->pos += filtered;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Flush pending filtered data from temp to the output buffer.
|
||||||
|
* Move the remaining mixture of possibly filtered and unfiltered
|
||||||
|
* data to the beginning of temp.
|
||||||
|
*/
|
||||||
|
static void bcj_flush(struct xz_dec_bcj *s, struct xz_buf *b)
|
||||||
|
{
|
||||||
|
size_t copy_size;
|
||||||
|
|
||||||
|
copy_size = min_t(size_t, s->temp.filtered, b->out_size - b->out_pos);
|
||||||
|
memcpy(b->out + b->out_pos, s->temp.buf, copy_size);
|
||||||
|
b->out_pos += copy_size;
|
||||||
|
|
||||||
|
s->temp.filtered -= copy_size;
|
||||||
|
s->temp.size -= copy_size;
|
||||||
|
memmove(s->temp.buf, s->temp.buf + copy_size, s->temp.size);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* The BCJ filter functions are primitive in sense that they process the
|
||||||
|
* data in chunks of 1-16 bytes. To hide this issue, this function does
|
||||||
|
* some buffering.
|
||||||
|
*/
|
||||||
|
XZ_EXTERN enum xz_ret xz_dec_bcj_run(struct xz_dec_bcj *s,
|
||||||
|
struct xz_dec_lzma2 *lzma2,
|
||||||
|
struct xz_buf *b)
|
||||||
|
{
|
||||||
|
size_t out_start;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Flush pending already filtered data to the output buffer. Return
|
||||||
|
* immediately if we couldn't flush everything, or if the next
|
||||||
|
* filter in the chain had already returned XZ_STREAM_END.
|
||||||
|
*/
|
||||||
|
if (s->temp.filtered > 0) {
|
||||||
|
bcj_flush(s, b);
|
||||||
|
if (s->temp.filtered > 0)
|
||||||
|
return XZ_OK;
|
||||||
|
|
||||||
|
if (s->ret == XZ_STREAM_END)
|
||||||
|
return XZ_STREAM_END;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If we have more output space than what is currently pending in
|
||||||
|
* temp, copy the unfiltered data from temp to the output buffer
|
||||||
|
* and try to fill the output buffer by decoding more data from the
|
||||||
|
* next filter in the chain. Apply the BCJ filter on the new data
|
||||||
|
* in the output buffer. If everything cannot be filtered, copy it
|
||||||
|
* to temp and rewind the output buffer position accordingly.
|
||||||
|
*
|
||||||
|
* This needs to be always run when temp.size == 0 to handle a special
|
||||||
|
* case where the output buffer is full and the next filter has no
|
||||||
|
* more output coming but hasn't returned XZ_STREAM_END yet.
|
||||||
|
*/
|
||||||
|
if (s->temp.size < b->out_size - b->out_pos || s->temp.size == 0) {
|
||||||
|
out_start = b->out_pos;
|
||||||
|
memcpy(b->out + b->out_pos, s->temp.buf, s->temp.size);
|
||||||
|
b->out_pos += s->temp.size;
|
||||||
|
|
||||||
|
s->ret = xz_dec_lzma2_run(lzma2, b);
|
||||||
|
if (s->ret != XZ_STREAM_END
|
||||||
|
&& (s->ret != XZ_OK || s->single_call))
|
||||||
|
return s->ret;
|
||||||
|
|
||||||
|
bcj_apply(s, b->out, &out_start, b->out_pos);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* As an exception, if the next filter returned XZ_STREAM_END,
|
||||||
|
* we can do that too, since the last few bytes that remain
|
||||||
|
* unfiltered are meant to remain unfiltered.
|
||||||
|
*/
|
||||||
|
if (s->ret == XZ_STREAM_END)
|
||||||
|
return XZ_STREAM_END;
|
||||||
|
|
||||||
|
s->temp.size = b->out_pos - out_start;
|
||||||
|
b->out_pos -= s->temp.size;
|
||||||
|
memcpy(s->temp.buf, b->out + b->out_pos, s->temp.size);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If there wasn't enough input to the next filter to fill
|
||||||
|
* the output buffer with unfiltered data, there's no point
|
||||||
|
* to try decoding more data to temp.
|
||||||
|
*/
|
||||||
|
if (b->out_pos + s->temp.size < b->out_size)
|
||||||
|
return XZ_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* We have unfiltered data in temp. If the output buffer isn't full
|
||||||
|
* yet, try to fill the temp buffer by decoding more data from the
|
||||||
|
* next filter. Apply the BCJ filter on temp. Then we hopefully can
|
||||||
|
* fill the actual output buffer by copying filtered data from temp.
|
||||||
|
* A mix of filtered and unfiltered data may be left in temp; it will
|
||||||
|
* be taken care on the next call to this function.
|
||||||
|
*/
|
||||||
|
if (b->out_pos < b->out_size) {
|
||||||
|
/* Make b->out{,_pos,_size} temporarily point to s->temp. */
|
||||||
|
s->out = b->out;
|
||||||
|
s->out_pos = b->out_pos;
|
||||||
|
s->out_size = b->out_size;
|
||||||
|
b->out = s->temp.buf;
|
||||||
|
b->out_pos = s->temp.size;
|
||||||
|
b->out_size = sizeof(s->temp.buf);
|
||||||
|
|
||||||
|
s->ret = xz_dec_lzma2_run(lzma2, b);
|
||||||
|
|
||||||
|
s->temp.size = b->out_pos;
|
||||||
|
b->out = s->out;
|
||||||
|
b->out_pos = s->out_pos;
|
||||||
|
b->out_size = s->out_size;
|
||||||
|
|
||||||
|
if (s->ret != XZ_OK && s->ret != XZ_STREAM_END)
|
||||||
|
return s->ret;
|
||||||
|
|
||||||
|
bcj_apply(s, s->temp.buf, &s->temp.filtered, s->temp.size);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If the next filter returned XZ_STREAM_END, we mark that
|
||||||
|
* everything is filtered, since the last unfiltered bytes
|
||||||
|
* of the stream are meant to be left as is.
|
||||||
|
*/
|
||||||
|
if (s->ret == XZ_STREAM_END)
|
||||||
|
s->temp.filtered = s->temp.size;
|
||||||
|
|
||||||
|
bcj_flush(s, b);
|
||||||
|
if (s->temp.filtered > 0)
|
||||||
|
return XZ_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
return s->ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
XZ_EXTERN struct xz_dec_bcj *xz_dec_bcj_create(bool single_call)
|
||||||
|
{
|
||||||
|
struct xz_dec_bcj *s = kmalloc(sizeof(*s), GFP_KERNEL);
|
||||||
|
if (s != NULL)
|
||||||
|
s->single_call = single_call;
|
||||||
|
|
||||||
|
return s;
|
||||||
|
}
|
||||||
|
|
||||||
|
XZ_EXTERN enum xz_ret xz_dec_bcj_reset(struct xz_dec_bcj *s, uint8_t id)
|
||||||
|
{
|
||||||
|
switch (id) {
|
||||||
|
#ifdef XZ_DEC_X86
|
||||||
|
case BCJ_X86:
|
||||||
|
#endif
|
||||||
|
#ifdef XZ_DEC_POWERPC
|
||||||
|
case BCJ_POWERPC:
|
||||||
|
#endif
|
||||||
|
#ifdef XZ_DEC_IA64
|
||||||
|
case BCJ_IA64:
|
||||||
|
#endif
|
||||||
|
#ifdef XZ_DEC_ARM
|
||||||
|
case BCJ_ARM:
|
||||||
|
#endif
|
||||||
|
#ifdef XZ_DEC_ARMTHUMB
|
||||||
|
case BCJ_ARMTHUMB:
|
||||||
|
#endif
|
||||||
|
#ifdef XZ_DEC_SPARC
|
||||||
|
case BCJ_SPARC:
|
||||||
|
#endif
|
||||||
|
#ifdef XZ_DEC_ARM64
|
||||||
|
case BCJ_ARM64:
|
||||||
|
#endif
|
||||||
|
#ifdef XZ_DEC_RISCV
|
||||||
|
case BCJ_RISCV:
|
||||||
|
#endif
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
/* Unsupported Filter ID */
|
||||||
|
return XZ_OPTIONS_ERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
s->type = id;
|
||||||
|
s->ret = XZ_OK;
|
||||||
|
s->pos = 0;
|
||||||
|
s->x86_prev_mask = 0;
|
||||||
|
s->temp.filtered = 0;
|
||||||
|
s->temp.size = 0;
|
||||||
|
|
||||||
|
return XZ_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
1345
android/app/src/main/cpp/xz/xz_dec_lzma2.c
Normal file
1345
android/app/src/main/cpp/xz/xz_dec_lzma2.c
Normal file
File diff suppressed because it is too large
Load Diff
984
android/app/src/main/cpp/xz/xz_dec_stream.c
Normal file
984
android/app/src/main/cpp/xz/xz_dec_stream.c
Normal file
@@ -0,0 +1,984 @@
|
|||||||
|
// SPDX-License-Identifier: 0BSD
|
||||||
|
|
||||||
|
/*
|
||||||
|
* .xz Stream decoder
|
||||||
|
*
|
||||||
|
* Author: Lasse Collin <lasse.collin@tukaani.org>
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "xz_private.h"
|
||||||
|
#include "xz_stream.h"
|
||||||
|
|
||||||
|
#ifdef XZ_USE_CRC64
|
||||||
|
# define IS_CRC64(check_type) ((check_type) == XZ_CHECK_CRC64)
|
||||||
|
#else
|
||||||
|
# define IS_CRC64(check_type) false
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef XZ_USE_SHA256
|
||||||
|
# define IS_SHA256(check_type) ((check_type) == XZ_CHECK_SHA256)
|
||||||
|
#else
|
||||||
|
# define IS_SHA256(check_type) false
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* Hash used to validate the Index field */
|
||||||
|
struct xz_dec_hash {
|
||||||
|
vli_type unpadded;
|
||||||
|
vli_type uncompressed;
|
||||||
|
uint32_t crc32;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct xz_dec {
|
||||||
|
/* Position in dec_main() */
|
||||||
|
enum {
|
||||||
|
SEQ_STREAM_HEADER,
|
||||||
|
SEQ_BLOCK_START,
|
||||||
|
SEQ_BLOCK_HEADER,
|
||||||
|
SEQ_BLOCK_UNCOMPRESS,
|
||||||
|
SEQ_BLOCK_PADDING,
|
||||||
|
SEQ_BLOCK_CHECK,
|
||||||
|
SEQ_INDEX,
|
||||||
|
SEQ_INDEX_PADDING,
|
||||||
|
SEQ_INDEX_CRC32,
|
||||||
|
SEQ_STREAM_FOOTER,
|
||||||
|
SEQ_STREAM_PADDING
|
||||||
|
} sequence;
|
||||||
|
|
||||||
|
/* Position in variable-length integers and Check fields */
|
||||||
|
uint32_t pos;
|
||||||
|
|
||||||
|
/* Variable-length integer decoded by dec_vli() */
|
||||||
|
vli_type vli;
|
||||||
|
|
||||||
|
/* Saved in_pos and out_pos */
|
||||||
|
size_t in_start;
|
||||||
|
size_t out_start;
|
||||||
|
|
||||||
|
#ifdef XZ_USE_CRC64
|
||||||
|
/* CRC32 or CRC64 value in Block or CRC32 value in Index */
|
||||||
|
uint64_t crc;
|
||||||
|
#else
|
||||||
|
/* CRC32 value in Block or Index */
|
||||||
|
uint32_t crc;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* Type of the integrity check calculated from uncompressed data */
|
||||||
|
enum xz_check check_type;
|
||||||
|
|
||||||
|
/* Operation mode */
|
||||||
|
enum xz_mode mode;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* True if the next call to xz_dec_run() is allowed to return
|
||||||
|
* XZ_BUF_ERROR.
|
||||||
|
*/
|
||||||
|
bool allow_buf_error;
|
||||||
|
|
||||||
|
/* Information stored in Block Header */
|
||||||
|
struct {
|
||||||
|
/*
|
||||||
|
* Value stored in the Compressed Size field, or
|
||||||
|
* VLI_UNKNOWN if Compressed Size is not present.
|
||||||
|
*/
|
||||||
|
vli_type compressed;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Value stored in the Uncompressed Size field, or
|
||||||
|
* VLI_UNKNOWN if Uncompressed Size is not present.
|
||||||
|
*/
|
||||||
|
vli_type uncompressed;
|
||||||
|
|
||||||
|
/* Size of the Block Header field */
|
||||||
|
uint32_t size;
|
||||||
|
} block_header;
|
||||||
|
|
||||||
|
/* Information collected when decoding Blocks */
|
||||||
|
struct {
|
||||||
|
/* Observed compressed size of the current Block */
|
||||||
|
vli_type compressed;
|
||||||
|
|
||||||
|
/* Observed uncompressed size of the current Block */
|
||||||
|
vli_type uncompressed;
|
||||||
|
|
||||||
|
/* Number of Blocks decoded so far */
|
||||||
|
vli_type count;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Hash calculated from the Block sizes. This is used to
|
||||||
|
* validate the Index field.
|
||||||
|
*/
|
||||||
|
struct xz_dec_hash hash;
|
||||||
|
} block;
|
||||||
|
|
||||||
|
/* Variables needed when verifying the Index field */
|
||||||
|
struct {
|
||||||
|
/* Position in dec_index() */
|
||||||
|
enum {
|
||||||
|
SEQ_INDEX_COUNT,
|
||||||
|
SEQ_INDEX_UNPADDED,
|
||||||
|
SEQ_INDEX_UNCOMPRESSED
|
||||||
|
} sequence;
|
||||||
|
|
||||||
|
/* Size of the Index in bytes */
|
||||||
|
vli_type size;
|
||||||
|
|
||||||
|
/* Number of Records (matches block.count in valid files) */
|
||||||
|
vli_type count;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Hash calculated from the Records (matches block.hash in
|
||||||
|
* valid files).
|
||||||
|
*/
|
||||||
|
struct xz_dec_hash hash;
|
||||||
|
} index;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Temporary buffer needed to hold Stream Header, Block Header,
|
||||||
|
* and Stream Footer. The Block Header is the biggest (1 KiB)
|
||||||
|
* so we reserve space according to that. buf[] has to be aligned
|
||||||
|
* to a multiple of four bytes; the size_t variables before it
|
||||||
|
* should guarantee this.
|
||||||
|
*/
|
||||||
|
struct {
|
||||||
|
size_t pos;
|
||||||
|
size_t size;
|
||||||
|
uint8_t buf[1024];
|
||||||
|
} temp;
|
||||||
|
|
||||||
|
struct xz_dec_lzma2 *lzma2;
|
||||||
|
|
||||||
|
#ifdef XZ_DEC_BCJ
|
||||||
|
struct xz_dec_bcj *bcj;
|
||||||
|
bool bcj_active;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef XZ_USE_SHA256
|
||||||
|
/*
|
||||||
|
* SHA-256 value in Block
|
||||||
|
*
|
||||||
|
* struct xz_sha256 is over a hundred bytes and it's only accessed
|
||||||
|
* from a few places. By putting the SHA-256 state near the end
|
||||||
|
* of struct xz_dec (somewhere after the "index" member) reduces
|
||||||
|
* code size at least on x86 and RISC-V. It's because the first bytes
|
||||||
|
* of the struct can be accessed with smaller instructions; the
|
||||||
|
* members that are accessed from many places should be at the top.
|
||||||
|
*/
|
||||||
|
struct xz_sha256 sha256;
|
||||||
|
#endif
|
||||||
|
};
|
||||||
|
|
||||||
|
#if defined(XZ_DEC_ANY_CHECK) || defined(XZ_USE_SHA256)
|
||||||
|
/* Sizes of the Check field with different Check IDs */
|
||||||
|
static const uint8_t check_sizes[16] = {
|
||||||
|
0,
|
||||||
|
4, 4, 4,
|
||||||
|
8, 8, 8,
|
||||||
|
16, 16, 16,
|
||||||
|
32, 32, 32,
|
||||||
|
64, 64, 64
|
||||||
|
};
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Fill s->temp by copying data starting from b->in[b->in_pos]. Caller
|
||||||
|
* must have set s->temp.pos and s->temp.size to indicate how much data
|
||||||
|
* we are supposed to copy into s->temp.buf. Return true once s->temp.pos
|
||||||
|
* has reached s->temp.size.
|
||||||
|
*/
|
||||||
|
static bool fill_temp(struct xz_dec *s, struct xz_buf *b)
|
||||||
|
{
|
||||||
|
size_t copy_size = min_t(size_t,
|
||||||
|
b->in_size - b->in_pos, s->temp.size - s->temp.pos);
|
||||||
|
|
||||||
|
memcpy(s->temp.buf + s->temp.pos, b->in + b->in_pos, copy_size);
|
||||||
|
b->in_pos += copy_size;
|
||||||
|
s->temp.pos += copy_size;
|
||||||
|
|
||||||
|
if (s->temp.pos == s->temp.size) {
|
||||||
|
s->temp.pos = 0;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Decode a variable-length integer (little-endian base-128 encoding) */
|
||||||
|
static enum xz_ret dec_vli(struct xz_dec *s, const uint8_t *in,
|
||||||
|
size_t *in_pos, size_t in_size)
|
||||||
|
{
|
||||||
|
uint8_t byte;
|
||||||
|
|
||||||
|
if (s->pos == 0)
|
||||||
|
s->vli = 0;
|
||||||
|
|
||||||
|
while (*in_pos < in_size) {
|
||||||
|
byte = in[*in_pos];
|
||||||
|
++*in_pos;
|
||||||
|
|
||||||
|
s->vli |= (vli_type)(byte & 0x7F) << s->pos;
|
||||||
|
|
||||||
|
if ((byte & 0x80) == 0) {
|
||||||
|
/* Don't allow non-minimal encodings. */
|
||||||
|
if (byte == 0 && s->pos != 0)
|
||||||
|
return XZ_DATA_ERROR;
|
||||||
|
|
||||||
|
s->pos = 0;
|
||||||
|
return XZ_STREAM_END;
|
||||||
|
}
|
||||||
|
|
||||||
|
s->pos += 7;
|
||||||
|
if (s->pos == 7 * VLI_BYTES_MAX)
|
||||||
|
return XZ_DATA_ERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
return XZ_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Decode the Compressed Data field from a Block. Update and validate
|
||||||
|
* the observed compressed and uncompressed sizes of the Block so that
|
||||||
|
* they don't exceed the values possibly stored in the Block Header
|
||||||
|
* (validation assumes that no integer overflow occurs, since vli_type
|
||||||
|
* is normally uint64_t). Update the CRC32 or CRC64 value if presence of
|
||||||
|
* the CRC32 or CRC64 field was indicated in Stream Header.
|
||||||
|
*
|
||||||
|
* Once the decoding is finished, validate that the observed sizes match
|
||||||
|
* the sizes possibly stored in the Block Header. Update the hash and
|
||||||
|
* Block count, which are later used to validate the Index field.
|
||||||
|
*/
|
||||||
|
static enum xz_ret dec_block(struct xz_dec *s, struct xz_buf *b)
|
||||||
|
{
|
||||||
|
enum xz_ret ret;
|
||||||
|
|
||||||
|
s->in_start = b->in_pos;
|
||||||
|
s->out_start = b->out_pos;
|
||||||
|
|
||||||
|
#ifdef XZ_DEC_BCJ
|
||||||
|
if (s->bcj_active)
|
||||||
|
ret = xz_dec_bcj_run(s->bcj, s->lzma2, b);
|
||||||
|
else
|
||||||
|
#endif
|
||||||
|
ret = xz_dec_lzma2_run(s->lzma2, b);
|
||||||
|
|
||||||
|
s->block.compressed += b->in_pos - s->in_start;
|
||||||
|
s->block.uncompressed += b->out_pos - s->out_start;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* There is no need to separately check for VLI_UNKNOWN, since
|
||||||
|
* the observed sizes are always smaller than VLI_UNKNOWN.
|
||||||
|
*/
|
||||||
|
if (s->block.compressed > s->block_header.compressed
|
||||||
|
|| s->block.uncompressed
|
||||||
|
> s->block_header.uncompressed)
|
||||||
|
return XZ_DATA_ERROR;
|
||||||
|
|
||||||
|
if (s->check_type == XZ_CHECK_CRC32)
|
||||||
|
s->crc = xz_crc32(b->out + s->out_start,
|
||||||
|
b->out_pos - s->out_start, s->crc);
|
||||||
|
#ifdef XZ_USE_CRC64
|
||||||
|
else if (s->check_type == XZ_CHECK_CRC64)
|
||||||
|
s->crc = xz_crc64(b->out + s->out_start,
|
||||||
|
b->out_pos - s->out_start, s->crc);
|
||||||
|
#endif
|
||||||
|
#ifdef XZ_USE_SHA256
|
||||||
|
else if (s->check_type == XZ_CHECK_SHA256)
|
||||||
|
xz_sha256_update(b->out + s->out_start,
|
||||||
|
b->out_pos - s->out_start, &s->sha256);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
if (ret == XZ_STREAM_END) {
|
||||||
|
if (s->block_header.compressed != VLI_UNKNOWN
|
||||||
|
&& s->block_header.compressed
|
||||||
|
!= s->block.compressed)
|
||||||
|
return XZ_DATA_ERROR;
|
||||||
|
|
||||||
|
if (s->block_header.uncompressed != VLI_UNKNOWN
|
||||||
|
&& s->block_header.uncompressed
|
||||||
|
!= s->block.uncompressed)
|
||||||
|
return XZ_DATA_ERROR;
|
||||||
|
|
||||||
|
s->block.hash.unpadded += s->block_header.size
|
||||||
|
+ s->block.compressed;
|
||||||
|
|
||||||
|
#if defined(XZ_DEC_ANY_CHECK) || defined(XZ_USE_SHA256)
|
||||||
|
s->block.hash.unpadded += check_sizes[s->check_type];
|
||||||
|
#else
|
||||||
|
if (s->check_type == XZ_CHECK_CRC32)
|
||||||
|
s->block.hash.unpadded += 4;
|
||||||
|
else if (IS_CRC64(s->check_type))
|
||||||
|
s->block.hash.unpadded += 8;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
s->block.hash.uncompressed += s->block.uncompressed;
|
||||||
|
s->block.hash.crc32 = xz_crc32(
|
||||||
|
(const uint8_t *)&s->block.hash,
|
||||||
|
sizeof(s->block.hash), s->block.hash.crc32);
|
||||||
|
|
||||||
|
++s->block.count;
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Update the Index size and the CRC32 value. */
|
||||||
|
static void index_update(struct xz_dec *s, const struct xz_buf *b)
|
||||||
|
{
|
||||||
|
size_t in_used = b->in_pos - s->in_start;
|
||||||
|
s->index.size += in_used;
|
||||||
|
s->crc = xz_crc32(b->in + s->in_start, in_used, s->crc);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Decode the Number of Records, Unpadded Size, and Uncompressed Size
|
||||||
|
* fields from the Index field. That is, Index Padding and CRC32 are not
|
||||||
|
* decoded by this function.
|
||||||
|
*
|
||||||
|
* This can return XZ_OK (more input needed), XZ_STREAM_END (everything
|
||||||
|
* successfully decoded), or XZ_DATA_ERROR (input is corrupt).
|
||||||
|
*/
|
||||||
|
static enum xz_ret dec_index(struct xz_dec *s, struct xz_buf *b)
|
||||||
|
{
|
||||||
|
enum xz_ret ret;
|
||||||
|
|
||||||
|
do {
|
||||||
|
ret = dec_vli(s, b->in, &b->in_pos, b->in_size);
|
||||||
|
if (ret != XZ_STREAM_END) {
|
||||||
|
index_update(s, b);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (s->index.sequence) {
|
||||||
|
case SEQ_INDEX_COUNT:
|
||||||
|
s->index.count = s->vli;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Validate that the Number of Records field
|
||||||
|
* indicates the same number of Records as
|
||||||
|
* there were Blocks in the Stream.
|
||||||
|
*/
|
||||||
|
if (s->index.count != s->block.count)
|
||||||
|
return XZ_DATA_ERROR;
|
||||||
|
|
||||||
|
s->index.sequence = SEQ_INDEX_UNPADDED;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case SEQ_INDEX_UNPADDED:
|
||||||
|
s->index.hash.unpadded += s->vli;
|
||||||
|
s->index.sequence = SEQ_INDEX_UNCOMPRESSED;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case SEQ_INDEX_UNCOMPRESSED:
|
||||||
|
s->index.hash.uncompressed += s->vli;
|
||||||
|
s->index.hash.crc32 = xz_crc32(
|
||||||
|
(const uint8_t *)&s->index.hash,
|
||||||
|
sizeof(s->index.hash),
|
||||||
|
s->index.hash.crc32);
|
||||||
|
--s->index.count;
|
||||||
|
s->index.sequence = SEQ_INDEX_UNPADDED;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
} while (s->index.count > 0);
|
||||||
|
|
||||||
|
return XZ_STREAM_END;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Validate that the next four or eight input bytes match the value
|
||||||
|
* of s->crc. s->pos must be zero when starting to validate the first byte.
|
||||||
|
* The "bits" argument allows using the same code for both CRC32 and CRC64.
|
||||||
|
*/
|
||||||
|
static enum xz_ret crc_validate(struct xz_dec *s, struct xz_buf *b,
|
||||||
|
uint32_t bits)
|
||||||
|
{
|
||||||
|
do {
|
||||||
|
if (b->in_pos == b->in_size)
|
||||||
|
return XZ_OK;
|
||||||
|
|
||||||
|
if (((s->crc >> s->pos) & 0xFF) != b->in[b->in_pos++])
|
||||||
|
return XZ_DATA_ERROR;
|
||||||
|
|
||||||
|
s->pos += 8;
|
||||||
|
|
||||||
|
} while (s->pos < bits);
|
||||||
|
|
||||||
|
s->crc = 0;
|
||||||
|
s->pos = 0;
|
||||||
|
|
||||||
|
return XZ_STREAM_END;
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifdef XZ_DEC_ANY_CHECK
|
||||||
|
/*
|
||||||
|
* Skip over the Check field when the Check ID is not supported.
|
||||||
|
* Returns true once the whole Check field has been skipped over.
|
||||||
|
*/
|
||||||
|
static bool check_skip(struct xz_dec *s, struct xz_buf *b)
|
||||||
|
{
|
||||||
|
while (s->pos < check_sizes[s->check_type]) {
|
||||||
|
if (b->in_pos == b->in_size)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
++b->in_pos;
|
||||||
|
++s->pos;
|
||||||
|
}
|
||||||
|
|
||||||
|
s->pos = 0;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* Decode the Stream Header field (the first 12 bytes of the .xz Stream). */
|
||||||
|
static enum xz_ret dec_stream_header(struct xz_dec *s)
|
||||||
|
{
|
||||||
|
if (!memeq(s->temp.buf, HEADER_MAGIC, HEADER_MAGIC_SIZE))
|
||||||
|
return XZ_FORMAT_ERROR;
|
||||||
|
|
||||||
|
if (xz_crc32(s->temp.buf + HEADER_MAGIC_SIZE, 2, 0)
|
||||||
|
!= get_le32(s->temp.buf + HEADER_MAGIC_SIZE + 2))
|
||||||
|
return XZ_DATA_ERROR;
|
||||||
|
|
||||||
|
if (s->temp.buf[HEADER_MAGIC_SIZE] != 0)
|
||||||
|
return XZ_OPTIONS_ERROR;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Of integrity checks, we support none (Check ID = 0),
|
||||||
|
* CRC32 (Check ID = 1), and optionally CRC64 (Check ID = 4).
|
||||||
|
* However, if XZ_DEC_ANY_CHECK is defined, we will accept other
|
||||||
|
* check types too, but then the check won't be verified and
|
||||||
|
* a warning (XZ_UNSUPPORTED_CHECK) will be given.
|
||||||
|
*/
|
||||||
|
if (s->temp.buf[HEADER_MAGIC_SIZE + 1] > XZ_CHECK_MAX)
|
||||||
|
return XZ_OPTIONS_ERROR;
|
||||||
|
|
||||||
|
s->check_type = s->temp.buf[HEADER_MAGIC_SIZE + 1];
|
||||||
|
|
||||||
|
if (s->check_type > XZ_CHECK_CRC32 && !IS_CRC64(s->check_type)
|
||||||
|
&& !IS_SHA256(s->check_type)) {
|
||||||
|
#ifdef XZ_DEC_ANY_CHECK
|
||||||
|
return XZ_UNSUPPORTED_CHECK;
|
||||||
|
#else
|
||||||
|
return XZ_OPTIONS_ERROR;
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
return XZ_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Decode the Stream Footer field (the last 12 bytes of the .xz Stream) */
|
||||||
|
static enum xz_ret dec_stream_footer(struct xz_dec *s)
|
||||||
|
{
|
||||||
|
if (!memeq(s->temp.buf + 10, FOOTER_MAGIC, FOOTER_MAGIC_SIZE))
|
||||||
|
return XZ_DATA_ERROR;
|
||||||
|
|
||||||
|
if (xz_crc32(s->temp.buf + 4, 6, 0) != get_le32(s->temp.buf))
|
||||||
|
return XZ_DATA_ERROR;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Validate Backward Size. Note that we never added the size of the
|
||||||
|
* Index CRC32 field to s->index.size, thus we use s->index.size / 4
|
||||||
|
* instead of s->index.size / 4 - 1.
|
||||||
|
*/
|
||||||
|
if ((s->index.size >> 2) != get_le32(s->temp.buf + 4))
|
||||||
|
return XZ_DATA_ERROR;
|
||||||
|
|
||||||
|
if (s->temp.buf[8] != 0 || s->temp.buf[9] != s->check_type)
|
||||||
|
return XZ_DATA_ERROR;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Use XZ_STREAM_END instead of XZ_OK to be more convenient
|
||||||
|
* for the caller.
|
||||||
|
*/
|
||||||
|
return XZ_STREAM_END;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Decode the Block Header and initialize the filter chain. */
|
||||||
|
static enum xz_ret dec_block_header(struct xz_dec *s)
|
||||||
|
{
|
||||||
|
enum xz_ret ret;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Validate the CRC32. We know that the temp buffer is at least
|
||||||
|
* eight bytes so this is safe.
|
||||||
|
*/
|
||||||
|
s->temp.size -= 4;
|
||||||
|
if (xz_crc32(s->temp.buf, s->temp.size, 0)
|
||||||
|
!= get_le32(s->temp.buf + s->temp.size))
|
||||||
|
return XZ_DATA_ERROR;
|
||||||
|
|
||||||
|
s->temp.pos = 2;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Catch unsupported Block Flags. We support only one or two filters
|
||||||
|
* in the chain, so we catch that with the same test.
|
||||||
|
*/
|
||||||
|
#ifdef XZ_DEC_BCJ
|
||||||
|
if (s->temp.buf[1] & 0x3E)
|
||||||
|
#else
|
||||||
|
if (s->temp.buf[1] & 0x3F)
|
||||||
|
#endif
|
||||||
|
return XZ_OPTIONS_ERROR;
|
||||||
|
|
||||||
|
/* Compressed Size */
|
||||||
|
if (s->temp.buf[1] & 0x40) {
|
||||||
|
if (dec_vli(s, s->temp.buf, &s->temp.pos, s->temp.size)
|
||||||
|
!= XZ_STREAM_END)
|
||||||
|
return XZ_DATA_ERROR;
|
||||||
|
|
||||||
|
s->block_header.compressed = s->vli;
|
||||||
|
} else {
|
||||||
|
s->block_header.compressed = VLI_UNKNOWN;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Uncompressed Size */
|
||||||
|
if (s->temp.buf[1] & 0x80) {
|
||||||
|
if (dec_vli(s, s->temp.buf, &s->temp.pos, s->temp.size)
|
||||||
|
!= XZ_STREAM_END)
|
||||||
|
return XZ_DATA_ERROR;
|
||||||
|
|
||||||
|
s->block_header.uncompressed = s->vli;
|
||||||
|
} else {
|
||||||
|
s->block_header.uncompressed = VLI_UNKNOWN;
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifdef XZ_DEC_BCJ
|
||||||
|
/* If there are two filters, the first one must be a BCJ filter. */
|
||||||
|
s->bcj_active = s->temp.buf[1] & 0x01;
|
||||||
|
if (s->bcj_active) {
|
||||||
|
if (s->temp.size - s->temp.pos < 2)
|
||||||
|
return XZ_OPTIONS_ERROR;
|
||||||
|
|
||||||
|
ret = xz_dec_bcj_reset(s->bcj, s->temp.buf[s->temp.pos++]);
|
||||||
|
if (ret != XZ_OK)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* We don't support custom start offset,
|
||||||
|
* so Size of Properties must be zero.
|
||||||
|
*/
|
||||||
|
if (s->temp.buf[s->temp.pos++] != 0x00)
|
||||||
|
return XZ_OPTIONS_ERROR;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* Valid Filter Flags always take at least two bytes. */
|
||||||
|
if (s->temp.size - s->temp.pos < 2)
|
||||||
|
return XZ_DATA_ERROR;
|
||||||
|
|
||||||
|
/* Filter ID = LZMA2 */
|
||||||
|
if (s->temp.buf[s->temp.pos++] != 0x21)
|
||||||
|
return XZ_OPTIONS_ERROR;
|
||||||
|
|
||||||
|
/* Size of Properties = 1-byte Filter Properties */
|
||||||
|
if (s->temp.buf[s->temp.pos++] != 0x01)
|
||||||
|
return XZ_OPTIONS_ERROR;
|
||||||
|
|
||||||
|
/* Filter Properties contains LZMA2 dictionary size. */
|
||||||
|
if (s->temp.size - s->temp.pos < 1)
|
||||||
|
return XZ_DATA_ERROR;
|
||||||
|
|
||||||
|
ret = xz_dec_lzma2_reset(s->lzma2, s->temp.buf[s->temp.pos++]);
|
||||||
|
if (ret != XZ_OK)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
/* The rest must be Header Padding. */
|
||||||
|
while (s->temp.pos < s->temp.size)
|
||||||
|
if (s->temp.buf[s->temp.pos++] != 0x00)
|
||||||
|
return XZ_OPTIONS_ERROR;
|
||||||
|
|
||||||
|
s->temp.pos = 0;
|
||||||
|
s->block.compressed = 0;
|
||||||
|
s->block.uncompressed = 0;
|
||||||
|
|
||||||
|
return XZ_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
static enum xz_ret dec_main(struct xz_dec *s, struct xz_buf *b)
|
||||||
|
{
|
||||||
|
enum xz_ret ret;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Store the start position for the case when we are in the middle
|
||||||
|
* of the Index field.
|
||||||
|
*/
|
||||||
|
s->in_start = b->in_pos;
|
||||||
|
|
||||||
|
while (true) {
|
||||||
|
switch (s->sequence) {
|
||||||
|
case SEQ_STREAM_HEADER:
|
||||||
|
/*
|
||||||
|
* Stream Header is copied to s->temp, and then
|
||||||
|
* decoded from there. This way if the caller
|
||||||
|
* gives us only little input at a time, we can
|
||||||
|
* still keep the Stream Header decoding code
|
||||||
|
* simple. Similar approach is used in many places
|
||||||
|
* in this file.
|
||||||
|
*/
|
||||||
|
if (!fill_temp(s, b))
|
||||||
|
return XZ_OK;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If dec_stream_header() returns
|
||||||
|
* XZ_UNSUPPORTED_CHECK, it is still possible
|
||||||
|
* to continue decoding if working in multi-call
|
||||||
|
* mode. Thus, update s->sequence before calling
|
||||||
|
* dec_stream_header().
|
||||||
|
*/
|
||||||
|
s->sequence = SEQ_BLOCK_START;
|
||||||
|
|
||||||
|
ret = dec_stream_header(s);
|
||||||
|
if (ret != XZ_OK)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
fallthrough;
|
||||||
|
|
||||||
|
case SEQ_BLOCK_START:
|
||||||
|
/* We need one byte of input to continue. */
|
||||||
|
if (b->in_pos == b->in_size)
|
||||||
|
return XZ_OK;
|
||||||
|
|
||||||
|
/* See if this is the beginning of the Index field. */
|
||||||
|
if (b->in[b->in_pos] == 0) {
|
||||||
|
s->in_start = b->in_pos++;
|
||||||
|
s->sequence = SEQ_INDEX;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Calculate the size of the Block Header and
|
||||||
|
* prepare to decode it.
|
||||||
|
*/
|
||||||
|
s->block_header.size
|
||||||
|
= ((uint32_t)b->in[b->in_pos] + 1) * 4;
|
||||||
|
|
||||||
|
s->temp.size = s->block_header.size;
|
||||||
|
s->temp.pos = 0;
|
||||||
|
s->sequence = SEQ_BLOCK_HEADER;
|
||||||
|
|
||||||
|
fallthrough;
|
||||||
|
|
||||||
|
case SEQ_BLOCK_HEADER:
|
||||||
|
if (!fill_temp(s, b))
|
||||||
|
return XZ_OK;
|
||||||
|
|
||||||
|
ret = dec_block_header(s);
|
||||||
|
if (ret != XZ_OK)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
#ifdef XZ_USE_SHA256
|
||||||
|
if (s->check_type == XZ_CHECK_SHA256)
|
||||||
|
xz_sha256_reset(&s->sha256);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
s->sequence = SEQ_BLOCK_UNCOMPRESS;
|
||||||
|
|
||||||
|
fallthrough;
|
||||||
|
|
||||||
|
case SEQ_BLOCK_UNCOMPRESS:
|
||||||
|
ret = dec_block(s, b);
|
||||||
|
if (ret != XZ_STREAM_END)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
s->sequence = SEQ_BLOCK_PADDING;
|
||||||
|
|
||||||
|
fallthrough;
|
||||||
|
|
||||||
|
case SEQ_BLOCK_PADDING:
|
||||||
|
/*
|
||||||
|
* Size of Compressed Data + Block Padding
|
||||||
|
* must be a multiple of four. We don't need
|
||||||
|
* s->block.compressed for anything else
|
||||||
|
* anymore, so we use it here to test the size
|
||||||
|
* of the Block Padding field.
|
||||||
|
*/
|
||||||
|
while (s->block.compressed & 3) {
|
||||||
|
if (b->in_pos == b->in_size)
|
||||||
|
return XZ_OK;
|
||||||
|
|
||||||
|
if (b->in[b->in_pos++] != 0)
|
||||||
|
return XZ_DATA_ERROR;
|
||||||
|
|
||||||
|
++s->block.compressed;
|
||||||
|
}
|
||||||
|
|
||||||
|
s->sequence = SEQ_BLOCK_CHECK;
|
||||||
|
|
||||||
|
fallthrough;
|
||||||
|
|
||||||
|
case SEQ_BLOCK_CHECK:
|
||||||
|
if (s->check_type == XZ_CHECK_CRC32) {
|
||||||
|
ret = crc_validate(s, b, 32);
|
||||||
|
if (ret != XZ_STREAM_END)
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
else if (IS_CRC64(s->check_type)) {
|
||||||
|
ret = crc_validate(s, b, 64);
|
||||||
|
if (ret != XZ_STREAM_END)
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
#ifdef XZ_USE_SHA256
|
||||||
|
else if (s->check_type == XZ_CHECK_SHA256) {
|
||||||
|
s->temp.size = 32;
|
||||||
|
if (!fill_temp(s, b))
|
||||||
|
return XZ_OK;
|
||||||
|
|
||||||
|
if (!xz_sha256_validate(s->temp.buf,
|
||||||
|
&s->sha256))
|
||||||
|
return XZ_DATA_ERROR;
|
||||||
|
|
||||||
|
s->pos = 0;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
#ifdef XZ_DEC_ANY_CHECK
|
||||||
|
else if (!check_skip(s, b)) {
|
||||||
|
return XZ_OK;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
s->sequence = SEQ_BLOCK_START;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case SEQ_INDEX:
|
||||||
|
ret = dec_index(s, b);
|
||||||
|
if (ret != XZ_STREAM_END)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
s->sequence = SEQ_INDEX_PADDING;
|
||||||
|
|
||||||
|
fallthrough;
|
||||||
|
|
||||||
|
case SEQ_INDEX_PADDING:
|
||||||
|
while ((s->index.size + (b->in_pos - s->in_start))
|
||||||
|
& 3) {
|
||||||
|
if (b->in_pos == b->in_size) {
|
||||||
|
index_update(s, b);
|
||||||
|
return XZ_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (b->in[b->in_pos++] != 0)
|
||||||
|
return XZ_DATA_ERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Finish the CRC32 value and Index size. */
|
||||||
|
index_update(s, b);
|
||||||
|
|
||||||
|
/* Compare the hashes to validate the Index field. */
|
||||||
|
if (!memeq(&s->block.hash, &s->index.hash,
|
||||||
|
sizeof(s->block.hash)))
|
||||||
|
return XZ_DATA_ERROR;
|
||||||
|
|
||||||
|
s->sequence = SEQ_INDEX_CRC32;
|
||||||
|
|
||||||
|
fallthrough;
|
||||||
|
|
||||||
|
case SEQ_INDEX_CRC32:
|
||||||
|
ret = crc_validate(s, b, 32);
|
||||||
|
if (ret != XZ_STREAM_END)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
s->temp.size = STREAM_HEADER_SIZE;
|
||||||
|
s->sequence = SEQ_STREAM_FOOTER;
|
||||||
|
|
||||||
|
fallthrough;
|
||||||
|
|
||||||
|
case SEQ_STREAM_FOOTER:
|
||||||
|
if (!fill_temp(s, b))
|
||||||
|
return XZ_OK;
|
||||||
|
|
||||||
|
return dec_stream_footer(s);
|
||||||
|
|
||||||
|
case SEQ_STREAM_PADDING:
|
||||||
|
/* Never reached, only silencing a warning */
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Never reached */
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* xz_dec_run() is a wrapper for dec_main() to handle some special cases in
|
||||||
|
* multi-call and single-call decoding.
|
||||||
|
*
|
||||||
|
* In multi-call mode, we must return XZ_BUF_ERROR when it seems clear that we
|
||||||
|
* are not going to make any progress anymore. This is to prevent the caller
|
||||||
|
* from calling us infinitely when the input file is truncated or otherwise
|
||||||
|
* corrupt. Since zlib-style API allows that the caller fills the input buffer
|
||||||
|
* only when the decoder doesn't produce any new output, we have to be careful
|
||||||
|
* to avoid returning XZ_BUF_ERROR too easily: XZ_BUF_ERROR is returned only
|
||||||
|
* after the second consecutive call to xz_dec_run() that makes no progress.
|
||||||
|
*
|
||||||
|
* In single-call mode, if we couldn't decode everything and no error
|
||||||
|
* occurred, either the input is truncated or the output buffer is too small.
|
||||||
|
* Since we know that the last input byte never produces any output, we know
|
||||||
|
* that if all the input was consumed and decoding wasn't finished, the file
|
||||||
|
* must be corrupt. Otherwise the output buffer has to be too small or the
|
||||||
|
* file is corrupt in a way that decoding it produces too big output.
|
||||||
|
*
|
||||||
|
* If single-call decoding fails, we reset b->in_pos and b->out_pos back to
|
||||||
|
* their original values. This is because with some filter chains there won't
|
||||||
|
* be any valid uncompressed data in the output buffer unless the decoding
|
||||||
|
* actually succeeds (that's the price to pay of using the output buffer as
|
||||||
|
* the workspace).
|
||||||
|
*/
|
||||||
|
XZ_EXTERN enum xz_ret xz_dec_run(struct xz_dec *s, struct xz_buf *b)
|
||||||
|
{
|
||||||
|
size_t in_start;
|
||||||
|
size_t out_start;
|
||||||
|
enum xz_ret ret;
|
||||||
|
|
||||||
|
if (DEC_IS_SINGLE(s->mode))
|
||||||
|
xz_dec_reset(s);
|
||||||
|
|
||||||
|
in_start = b->in_pos;
|
||||||
|
out_start = b->out_pos;
|
||||||
|
ret = dec_main(s, b);
|
||||||
|
|
||||||
|
if (DEC_IS_SINGLE(s->mode)) {
|
||||||
|
if (ret == XZ_OK)
|
||||||
|
ret = b->in_pos == b->in_size
|
||||||
|
? XZ_DATA_ERROR : XZ_BUF_ERROR;
|
||||||
|
|
||||||
|
if (ret != XZ_STREAM_END) {
|
||||||
|
b->in_pos = in_start;
|
||||||
|
b->out_pos = out_start;
|
||||||
|
}
|
||||||
|
|
||||||
|
} else if (ret == XZ_OK && in_start == b->in_pos
|
||||||
|
&& out_start == b->out_pos) {
|
||||||
|
if (s->allow_buf_error)
|
||||||
|
ret = XZ_BUF_ERROR;
|
||||||
|
|
||||||
|
s->allow_buf_error = true;
|
||||||
|
} else {
|
||||||
|
s->allow_buf_error = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifdef XZ_DEC_CONCATENATED
|
||||||
|
XZ_EXTERN enum xz_ret xz_dec_catrun(struct xz_dec *s, struct xz_buf *b,
|
||||||
|
int finish)
|
||||||
|
{
|
||||||
|
enum xz_ret ret;
|
||||||
|
|
||||||
|
if (DEC_IS_SINGLE(s->mode)) {
|
||||||
|
xz_dec_reset(s);
|
||||||
|
finish = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
while (true) {
|
||||||
|
if (s->sequence == SEQ_STREAM_PADDING) {
|
||||||
|
/*
|
||||||
|
* Skip Stream Padding. Its size must be a multiple
|
||||||
|
* of four bytes which is tracked with s->pos.
|
||||||
|
*/
|
||||||
|
while (true) {
|
||||||
|
if (b->in_pos == b->in_size) {
|
||||||
|
/*
|
||||||
|
* Note that if we are repeatedly
|
||||||
|
* given no input and finish is false,
|
||||||
|
* we will keep returning XZ_OK even
|
||||||
|
* though no progress is being made.
|
||||||
|
* The lack of XZ_BUF_ERROR support
|
||||||
|
* isn't a problem here because a
|
||||||
|
* reasonable caller will eventually
|
||||||
|
* provide more input or set finish
|
||||||
|
* to true.
|
||||||
|
*/
|
||||||
|
if (!finish)
|
||||||
|
return XZ_OK;
|
||||||
|
|
||||||
|
if (s->pos != 0)
|
||||||
|
return XZ_DATA_ERROR;
|
||||||
|
|
||||||
|
return XZ_STREAM_END;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (b->in[b->in_pos] != 0x00) {
|
||||||
|
if (s->pos != 0)
|
||||||
|
return XZ_DATA_ERROR;
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
++b->in_pos;
|
||||||
|
s->pos = (s->pos + 1) & 3;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* More input remains. It should be a new Stream.
|
||||||
|
*
|
||||||
|
* In single-call mode xz_dec_run() will always call
|
||||||
|
* xz_dec_reset(). Thus, we need to do it here only
|
||||||
|
* in multi-call mode.
|
||||||
|
*/
|
||||||
|
if (DEC_IS_MULTI(s->mode))
|
||||||
|
xz_dec_reset(s);
|
||||||
|
}
|
||||||
|
|
||||||
|
ret = xz_dec_run(s, b);
|
||||||
|
|
||||||
|
if (ret != XZ_STREAM_END)
|
||||||
|
break;
|
||||||
|
|
||||||
|
s->sequence = SEQ_STREAM_PADDING;
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
XZ_EXTERN struct xz_dec *xz_dec_init(enum xz_mode mode, uint32_t dict_max)
|
||||||
|
{
|
||||||
|
struct xz_dec *s = kmalloc(sizeof(*s), GFP_KERNEL);
|
||||||
|
if (s == NULL)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
s->mode = mode;
|
||||||
|
|
||||||
|
#ifdef XZ_DEC_BCJ
|
||||||
|
s->bcj = xz_dec_bcj_create(DEC_IS_SINGLE(mode));
|
||||||
|
if (s->bcj == NULL)
|
||||||
|
goto error_bcj;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
s->lzma2 = xz_dec_lzma2_create(mode, dict_max);
|
||||||
|
if (s->lzma2 == NULL)
|
||||||
|
goto error_lzma2;
|
||||||
|
|
||||||
|
xz_dec_reset(s);
|
||||||
|
return s;
|
||||||
|
|
||||||
|
error_lzma2:
|
||||||
|
#ifdef XZ_DEC_BCJ
|
||||||
|
xz_dec_bcj_end(s->bcj);
|
||||||
|
error_bcj:
|
||||||
|
#endif
|
||||||
|
kfree(s);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
XZ_EXTERN void xz_dec_reset(struct xz_dec *s)
|
||||||
|
{
|
||||||
|
s->sequence = SEQ_STREAM_HEADER;
|
||||||
|
s->allow_buf_error = false;
|
||||||
|
s->pos = 0;
|
||||||
|
s->crc = 0;
|
||||||
|
memzero(&s->block, sizeof(s->block));
|
||||||
|
memzero(&s->index, sizeof(s->index));
|
||||||
|
s->temp.pos = 0;
|
||||||
|
s->temp.size = STREAM_HEADER_SIZE;
|
||||||
|
}
|
||||||
|
|
||||||
|
XZ_EXTERN void xz_dec_end(struct xz_dec *s)
|
||||||
|
{
|
||||||
|
if (s != NULL) {
|
||||||
|
xz_dec_lzma2_end(s->lzma2);
|
||||||
|
#ifdef XZ_DEC_BCJ
|
||||||
|
xz_dec_bcj_end(s->bcj);
|
||||||
|
#endif
|
||||||
|
kfree(s);
|
||||||
|
}
|
||||||
|
}
|
||||||
203
android/app/src/main/cpp/xz/xz_lzma2.h
Normal file
203
android/app/src/main/cpp/xz/xz_lzma2.h
Normal file
@@ -0,0 +1,203 @@
|
|||||||
|
/* SPDX-License-Identifier: 0BSD */
|
||||||
|
|
||||||
|
/*
|
||||||
|
* LZMA2 definitions
|
||||||
|
*
|
||||||
|
* Authors: Lasse Collin <lasse.collin@tukaani.org>
|
||||||
|
* Igor Pavlov <https://7-zip.org/>
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef XZ_LZMA2_H
|
||||||
|
#define XZ_LZMA2_H
|
||||||
|
|
||||||
|
/* Range coder constants */
|
||||||
|
#define RC_SHIFT_BITS 8
|
||||||
|
#define RC_TOP_BITS 24
|
||||||
|
#define RC_TOP_VALUE (1 << RC_TOP_BITS)
|
||||||
|
#define RC_BIT_MODEL_TOTAL_BITS 11
|
||||||
|
#define RC_BIT_MODEL_TOTAL (1 << RC_BIT_MODEL_TOTAL_BITS)
|
||||||
|
#define RC_MOVE_BITS 5
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Maximum number of position states. A position state is the lowest pb
|
||||||
|
* number of bits of the current uncompressed offset. In some places there
|
||||||
|
* are different sets of probabilities for different position states.
|
||||||
|
*/
|
||||||
|
#define POS_STATES_MAX (1 << 4)
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This enum is used to track which LZMA symbols have occurred most recently
|
||||||
|
* and in which order. This information is used to predict the next symbol.
|
||||||
|
*
|
||||||
|
* Symbols:
|
||||||
|
* - Literal: One 8-bit byte
|
||||||
|
* - Match: Repeat a chunk of data at some distance
|
||||||
|
* - Long repeat: Multi-byte match at a recently seen distance
|
||||||
|
* - Short repeat: One-byte repeat at a recently seen distance
|
||||||
|
*
|
||||||
|
* The symbol names are in from STATE_oldest_older_previous. REP means
|
||||||
|
* either short or long repeated match, and NONLIT means any non-literal.
|
||||||
|
*/
|
||||||
|
enum lzma_state {
|
||||||
|
STATE_LIT_LIT,
|
||||||
|
STATE_MATCH_LIT_LIT,
|
||||||
|
STATE_REP_LIT_LIT,
|
||||||
|
STATE_SHORTREP_LIT_LIT,
|
||||||
|
STATE_MATCH_LIT,
|
||||||
|
STATE_REP_LIT,
|
||||||
|
STATE_SHORTREP_LIT,
|
||||||
|
STATE_LIT_MATCH,
|
||||||
|
STATE_LIT_LONGREP,
|
||||||
|
STATE_LIT_SHORTREP,
|
||||||
|
STATE_NONLIT_MATCH,
|
||||||
|
STATE_NONLIT_REP
|
||||||
|
};
|
||||||
|
|
||||||
|
/* Total number of states */
|
||||||
|
#define STATES 12
|
||||||
|
|
||||||
|
/* The lowest 7 states indicate that the previous state was a literal. */
|
||||||
|
#define LIT_STATES 7
|
||||||
|
|
||||||
|
/* Indicate that the latest symbol was a literal. */
|
||||||
|
static inline void lzma_state_literal(enum lzma_state *state)
|
||||||
|
{
|
||||||
|
if (*state <= STATE_SHORTREP_LIT_LIT)
|
||||||
|
*state = STATE_LIT_LIT;
|
||||||
|
else if (*state <= STATE_LIT_SHORTREP)
|
||||||
|
*state -= 3;
|
||||||
|
else
|
||||||
|
*state -= 6;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Indicate that the latest symbol was a match. */
|
||||||
|
static inline void lzma_state_match(enum lzma_state *state)
|
||||||
|
{
|
||||||
|
*state = *state < LIT_STATES ? STATE_LIT_MATCH : STATE_NONLIT_MATCH;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Indicate that the latest state was a long repeated match. */
|
||||||
|
static inline void lzma_state_long_rep(enum lzma_state *state)
|
||||||
|
{
|
||||||
|
*state = *state < LIT_STATES ? STATE_LIT_LONGREP : STATE_NONLIT_REP;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Indicate that the latest symbol was a short match. */
|
||||||
|
static inline void lzma_state_short_rep(enum lzma_state *state)
|
||||||
|
{
|
||||||
|
*state = *state < LIT_STATES ? STATE_LIT_SHORTREP : STATE_NONLIT_REP;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Test if the previous symbol was a literal. */
|
||||||
|
static inline bool lzma_state_is_literal(enum lzma_state state)
|
||||||
|
{
|
||||||
|
return state < LIT_STATES;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Each literal coder is divided in three sections:
|
||||||
|
* - 0x001-0x0FF: Without match byte
|
||||||
|
* - 0x101-0x1FF: With match byte; match bit is 0
|
||||||
|
* - 0x201-0x2FF: With match byte; match bit is 1
|
||||||
|
*
|
||||||
|
* Match byte is used when the previous LZMA symbol was something else than
|
||||||
|
* a literal (that is, it was some kind of match).
|
||||||
|
*/
|
||||||
|
#define LITERAL_CODER_SIZE 0x300
|
||||||
|
|
||||||
|
/* Maximum number of literal coders */
|
||||||
|
#define LITERAL_CODERS_MAX (1 << 4)
|
||||||
|
|
||||||
|
/* Minimum length of a match is two bytes. */
|
||||||
|
#define MATCH_LEN_MIN 2
|
||||||
|
|
||||||
|
/* Match length is encoded with 4, 5, or 10 bits.
|
||||||
|
*
|
||||||
|
* Length Bits
|
||||||
|
* 2-9 4 = Choice=0 + 3 bits
|
||||||
|
* 10-17 5 = Choice=1 + Choice2=0 + 3 bits
|
||||||
|
* 18-273 10 = Choice=1 + Choice2=1 + 8 bits
|
||||||
|
*/
|
||||||
|
#define LEN_LOW_BITS 3
|
||||||
|
#define LEN_LOW_SYMBOLS (1 << LEN_LOW_BITS)
|
||||||
|
#define LEN_MID_BITS 3
|
||||||
|
#define LEN_MID_SYMBOLS (1 << LEN_MID_BITS)
|
||||||
|
#define LEN_HIGH_BITS 8
|
||||||
|
#define LEN_HIGH_SYMBOLS (1 << LEN_HIGH_BITS)
|
||||||
|
#define LEN_SYMBOLS (LEN_LOW_SYMBOLS + LEN_MID_SYMBOLS + LEN_HIGH_SYMBOLS)
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Maximum length of a match is 273 which is a result of the encoding
|
||||||
|
* described above.
|
||||||
|
*/
|
||||||
|
#define MATCH_LEN_MAX (MATCH_LEN_MIN + LEN_SYMBOLS - 1)
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Different sets of probabilities are used for match distances that have
|
||||||
|
* very short match length: Lengths of 2, 3, and 4 bytes have a separate
|
||||||
|
* set of probabilities for each length. The matches with longer length
|
||||||
|
* use a shared set of probabilities.
|
||||||
|
*/
|
||||||
|
#define DIST_STATES 4
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Get the index of the appropriate probability array for decoding
|
||||||
|
* the distance slot.
|
||||||
|
*/
|
||||||
|
static inline uint32_t lzma_get_dist_state(uint32_t len)
|
||||||
|
{
|
||||||
|
return len < DIST_STATES + MATCH_LEN_MIN
|
||||||
|
? len - MATCH_LEN_MIN : DIST_STATES - 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* The highest two bits of a 32-bit match distance are encoded using six bits.
|
||||||
|
* This six-bit value is called a distance slot. This way encoding a 32-bit
|
||||||
|
* value takes 6-36 bits, larger values taking more bits.
|
||||||
|
*/
|
||||||
|
#define DIST_SLOT_BITS 6
|
||||||
|
#define DIST_SLOTS (1 << DIST_SLOT_BITS)
|
||||||
|
|
||||||
|
/* Match distances up to 127 are fully encoded using probabilities. Since
|
||||||
|
* the highest two bits (distance slot) are always encoded using six bits,
|
||||||
|
* the distances 0-3 don't need any additional bits to encode, since the
|
||||||
|
* distance slot itself is the same as the actual distance. DIST_MODEL_START
|
||||||
|
* indicates the first distance slot where at least one additional bit is
|
||||||
|
* needed.
|
||||||
|
*/
|
||||||
|
#define DIST_MODEL_START 4
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Match distances greater than 127 are encoded in three pieces:
|
||||||
|
* - distance slot: the highest two bits
|
||||||
|
* - direct bits: 2-26 bits below the highest two bits
|
||||||
|
* - alignment bits: four lowest bits
|
||||||
|
*
|
||||||
|
* Direct bits don't use any probabilities.
|
||||||
|
*
|
||||||
|
* The distance slot value of 14 is for distances 128-191.
|
||||||
|
*/
|
||||||
|
#define DIST_MODEL_END 14
|
||||||
|
|
||||||
|
/* Distance slots that indicate a distance <= 127. */
|
||||||
|
#define FULL_DISTANCES_BITS (DIST_MODEL_END / 2)
|
||||||
|
#define FULL_DISTANCES (1 << FULL_DISTANCES_BITS)
|
||||||
|
|
||||||
|
/*
|
||||||
|
* For match distances greater than 127, only the highest two bits and the
|
||||||
|
* lowest four bits (alignment) is encoded using probabilities.
|
||||||
|
*/
|
||||||
|
#define ALIGN_BITS 4
|
||||||
|
#define ALIGN_SIZE (1 << ALIGN_BITS)
|
||||||
|
#define ALIGN_MASK (ALIGN_SIZE - 1)
|
||||||
|
|
||||||
|
/* Total number of all probability variables */
|
||||||
|
#define PROBS_TOTAL (1846 + LITERAL_CODERS_MAX * LITERAL_CODER_SIZE)
|
||||||
|
|
||||||
|
/*
|
||||||
|
* LZMA remembers the four most recent match distances. Reusing these
|
||||||
|
* distances tends to take less space than re-encoding the actual
|
||||||
|
* distance value.
|
||||||
|
*/
|
||||||
|
#define REPS 4
|
||||||
|
|
||||||
|
#endif
|
||||||
189
android/app/src/main/cpp/xz/xz_private.h
Normal file
189
android/app/src/main/cpp/xz/xz_private.h
Normal file
@@ -0,0 +1,189 @@
|
|||||||
|
/* SPDX-License-Identifier: 0BSD */
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Private includes and definitions
|
||||||
|
*
|
||||||
|
* Author: Lasse Collin <lasse.collin@tukaani.org>
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef XZ_PRIVATE_H
|
||||||
|
#define XZ_PRIVATE_H
|
||||||
|
|
||||||
|
#ifdef __KERNEL__
|
||||||
|
# include <linux/xz.h>
|
||||||
|
# include <linux/kernel.h>
|
||||||
|
# include <linux/unaligned.h>
|
||||||
|
/* XZ_PREBOOT may be defined only via decompress_unxz.c. */
|
||||||
|
# ifndef XZ_PREBOOT
|
||||||
|
# include <linux/slab.h>
|
||||||
|
# include <linux/vmalloc.h>
|
||||||
|
# include <linux/string.h>
|
||||||
|
# ifdef CONFIG_XZ_DEC_X86
|
||||||
|
# define XZ_DEC_X86
|
||||||
|
# endif
|
||||||
|
# ifdef CONFIG_XZ_DEC_POWERPC
|
||||||
|
# define XZ_DEC_POWERPC
|
||||||
|
# endif
|
||||||
|
# ifdef CONFIG_XZ_DEC_IA64
|
||||||
|
# define XZ_DEC_IA64
|
||||||
|
# endif
|
||||||
|
# ifdef CONFIG_XZ_DEC_ARM
|
||||||
|
# define XZ_DEC_ARM
|
||||||
|
# endif
|
||||||
|
# ifdef CONFIG_XZ_DEC_ARMTHUMB
|
||||||
|
# define XZ_DEC_ARMTHUMB
|
||||||
|
# endif
|
||||||
|
# ifdef CONFIG_XZ_DEC_SPARC
|
||||||
|
# define XZ_DEC_SPARC
|
||||||
|
# endif
|
||||||
|
# ifdef CONFIG_XZ_DEC_ARM64
|
||||||
|
# define XZ_DEC_ARM64
|
||||||
|
# endif
|
||||||
|
# ifdef CONFIG_XZ_DEC_RISCV
|
||||||
|
# define XZ_DEC_RISCV
|
||||||
|
# endif
|
||||||
|
# ifdef CONFIG_XZ_DEC_MICROLZMA
|
||||||
|
# define XZ_DEC_MICROLZMA
|
||||||
|
# endif
|
||||||
|
# define memeq(a, b, size) (memcmp(a, b, size) == 0)
|
||||||
|
# define memzero(buf, size) memset(buf, 0, size)
|
||||||
|
# endif
|
||||||
|
# define get_le32(p) le32_to_cpup((const uint32_t *)(p))
|
||||||
|
#else
|
||||||
|
/*
|
||||||
|
* For userspace builds, use a separate header to define the required
|
||||||
|
* macros and functions. This makes it easier to adapt the code into
|
||||||
|
* different environments and avoids clutter in the Linux kernel tree.
|
||||||
|
*/
|
||||||
|
# include "xz_config.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* If no specific decoding mode is requested, enable support for all modes. */
|
||||||
|
#if !defined(XZ_DEC_SINGLE) && !defined(XZ_DEC_PREALLOC) \
|
||||||
|
&& !defined(XZ_DEC_DYNALLOC)
|
||||||
|
# define XZ_DEC_SINGLE
|
||||||
|
# define XZ_DEC_PREALLOC
|
||||||
|
# define XZ_DEC_DYNALLOC
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/*
|
||||||
|
* The DEC_IS_foo(mode) macros are used in "if" statements. If only some
|
||||||
|
* of the supported modes are enabled, these macros will evaluate to true or
|
||||||
|
* false at compile time and thus allow the compiler to omit unneeded code.
|
||||||
|
*/
|
||||||
|
#ifdef XZ_DEC_SINGLE
|
||||||
|
# define DEC_IS_SINGLE(mode) ((mode) == XZ_SINGLE)
|
||||||
|
#else
|
||||||
|
# define DEC_IS_SINGLE(mode) (false)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef XZ_DEC_PREALLOC
|
||||||
|
# define DEC_IS_PREALLOC(mode) ((mode) == XZ_PREALLOC)
|
||||||
|
#else
|
||||||
|
# define DEC_IS_PREALLOC(mode) (false)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef XZ_DEC_DYNALLOC
|
||||||
|
# define DEC_IS_DYNALLOC(mode) ((mode) == XZ_DYNALLOC)
|
||||||
|
#else
|
||||||
|
# define DEC_IS_DYNALLOC(mode) (false)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if !defined(XZ_DEC_SINGLE)
|
||||||
|
# define DEC_IS_MULTI(mode) (true)
|
||||||
|
#elif defined(XZ_DEC_PREALLOC) || defined(XZ_DEC_DYNALLOC)
|
||||||
|
# define DEC_IS_MULTI(mode) ((mode) != XZ_SINGLE)
|
||||||
|
#else
|
||||||
|
# define DEC_IS_MULTI(mode) (false)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If any of the BCJ filter decoders are wanted, define XZ_DEC_BCJ.
|
||||||
|
* XZ_DEC_BCJ is used to enable generic support for BCJ decoders.
|
||||||
|
*/
|
||||||
|
#ifndef XZ_DEC_BCJ
|
||||||
|
# if defined(XZ_DEC_X86) || defined(XZ_DEC_POWERPC) \
|
||||||
|
|| defined(XZ_DEC_IA64) \
|
||||||
|
|| defined(XZ_DEC_ARM) || defined(XZ_DEC_ARMTHUMB) \
|
||||||
|
|| defined(XZ_DEC_SPARC) || defined(XZ_DEC_ARM64) \
|
||||||
|
|| defined(XZ_DEC_RISCV)
|
||||||
|
# define XZ_DEC_BCJ
|
||||||
|
# endif
|
||||||
|
#endif
|
||||||
|
|
||||||
|
struct xz_sha256 {
|
||||||
|
/* Buffered input data */
|
||||||
|
uint8_t data[64];
|
||||||
|
|
||||||
|
/* Internal state and the final hash value */
|
||||||
|
uint32_t state[8];
|
||||||
|
|
||||||
|
/* Size of the input data */
|
||||||
|
uint64_t size;
|
||||||
|
};
|
||||||
|
|
||||||
|
/* Reset the SHA-256 state to prepare for a new calculation. */
|
||||||
|
XZ_EXTERN void xz_sha256_reset(struct xz_sha256 *s);
|
||||||
|
|
||||||
|
/* Update the SHA-256 state with new data. */
|
||||||
|
XZ_EXTERN void xz_sha256_update(const uint8_t *buf, size_t size,
|
||||||
|
struct xz_sha256 *s);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Finish the SHA-256 calculation. Compare the result with the first 32 bytes
|
||||||
|
* from buf. Return true if the values are equal and false if they aren't.
|
||||||
|
*/
|
||||||
|
XZ_EXTERN bool xz_sha256_validate(const uint8_t *buf, struct xz_sha256 *s);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Allocate memory for LZMA2 decoder. xz_dec_lzma2_reset() must be used
|
||||||
|
* before calling xz_dec_lzma2_run().
|
||||||
|
*/
|
||||||
|
XZ_EXTERN struct xz_dec_lzma2 *xz_dec_lzma2_create(enum xz_mode mode,
|
||||||
|
uint32_t dict_max);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Decode the LZMA2 properties (one byte) and reset the decoder. Return
|
||||||
|
* XZ_OK on success, XZ_MEMLIMIT_ERROR if the preallocated dictionary is not
|
||||||
|
* big enough, and XZ_OPTIONS_ERROR if props indicates something that this
|
||||||
|
* decoder doesn't support.
|
||||||
|
*/
|
||||||
|
XZ_EXTERN enum xz_ret xz_dec_lzma2_reset(struct xz_dec_lzma2 *s,
|
||||||
|
uint8_t props);
|
||||||
|
|
||||||
|
/* Decode raw LZMA2 stream from b->in to b->out. */
|
||||||
|
XZ_EXTERN enum xz_ret xz_dec_lzma2_run(struct xz_dec_lzma2 *s,
|
||||||
|
struct xz_buf *b);
|
||||||
|
|
||||||
|
/* Free the memory allocated for the LZMA2 decoder. */
|
||||||
|
XZ_EXTERN void xz_dec_lzma2_end(struct xz_dec_lzma2 *s);
|
||||||
|
|
||||||
|
#ifdef XZ_DEC_BCJ
|
||||||
|
/*
|
||||||
|
* Allocate memory for BCJ decoders. xz_dec_bcj_reset() must be used before
|
||||||
|
* calling xz_dec_bcj_run().
|
||||||
|
*/
|
||||||
|
XZ_EXTERN struct xz_dec_bcj *xz_dec_bcj_create(bool single_call);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Decode the Filter ID of a BCJ filter. This implementation doesn't
|
||||||
|
* support custom start offsets, so no decoding of Filter Properties
|
||||||
|
* is needed. Returns XZ_OK if the given Filter ID is supported.
|
||||||
|
* Otherwise XZ_OPTIONS_ERROR is returned.
|
||||||
|
*/
|
||||||
|
XZ_EXTERN enum xz_ret xz_dec_bcj_reset(struct xz_dec_bcj *s, uint8_t id);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Decode raw BCJ + LZMA2 stream. This must be used only if there actually is
|
||||||
|
* a BCJ filter in the chain. If the chain has only LZMA2, xz_dec_lzma2_run()
|
||||||
|
* must be called directly.
|
||||||
|
*/
|
||||||
|
XZ_EXTERN enum xz_ret xz_dec_bcj_run(struct xz_dec_bcj *s,
|
||||||
|
struct xz_dec_lzma2 *lzma2,
|
||||||
|
struct xz_buf *b);
|
||||||
|
|
||||||
|
/* Free the memory allocated for the BCJ filters. */
|
||||||
|
#define xz_dec_bcj_end(s) kfree(s)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif
|
||||||
182
android/app/src/main/cpp/xz/xz_sha256.c
Normal file
182
android/app/src/main/cpp/xz/xz_sha256.c
Normal file
@@ -0,0 +1,182 @@
|
|||||||
|
// SPDX-License-Identifier: 0BSD
|
||||||
|
|
||||||
|
/*
|
||||||
|
* SHA-256
|
||||||
|
*
|
||||||
|
* This is based on the XZ Utils version which is based public domain code
|
||||||
|
* from Crypto++ Library 5.5.1 released in 2007: https://www.cryptopp.com/
|
||||||
|
*
|
||||||
|
* Authors: Wei Dai
|
||||||
|
* Lasse Collin <lasse.collin@tukaani.org>
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "xz_private.h"
|
||||||
|
|
||||||
|
static inline uint32_t
|
||||||
|
rotr_32(uint32_t num, unsigned amount)
|
||||||
|
{
|
||||||
|
return (num >> amount) | (num << (32 - amount));
|
||||||
|
}
|
||||||
|
|
||||||
|
#define blk0(i) (W[i] = get_be32(&data[4 * i]))
|
||||||
|
#define blk2(i) (W[i & 15] += s1(W[(i - 2) & 15]) + W[(i - 7) & 15] \
|
||||||
|
+ s0(W[(i - 15) & 15]))
|
||||||
|
|
||||||
|
#define Ch(x, y, z) (z ^ (x & (y ^ z)))
|
||||||
|
#define Maj(x, y, z) ((x & (y ^ z)) + (y & z))
|
||||||
|
|
||||||
|
#define a(i) T[(0 - i) & 7]
|
||||||
|
#define b(i) T[(1 - i) & 7]
|
||||||
|
#define c(i) T[(2 - i) & 7]
|
||||||
|
#define d(i) T[(3 - i) & 7]
|
||||||
|
#define e(i) T[(4 - i) & 7]
|
||||||
|
#define f(i) T[(5 - i) & 7]
|
||||||
|
#define g(i) T[(6 - i) & 7]
|
||||||
|
#define h(i) T[(7 - i) & 7]
|
||||||
|
|
||||||
|
#define R(i, j, blk) \
|
||||||
|
h(i) += S1(e(i)) + Ch(e(i), f(i), g(i)) + SHA256_K[i + j] + blk; \
|
||||||
|
d(i) += h(i); \
|
||||||
|
h(i) += S0(a(i)) + Maj(a(i), b(i), c(i))
|
||||||
|
#define R0(i) R(i, 0, blk0(i))
|
||||||
|
#define R2(i) R(i, j, blk2(i))
|
||||||
|
|
||||||
|
#define S0(x) rotr_32(x ^ rotr_32(x ^ rotr_32(x, 9), 11), 2)
|
||||||
|
#define S1(x) rotr_32(x ^ rotr_32(x ^ rotr_32(x, 14), 5), 6)
|
||||||
|
#define s0(x) (rotr_32(x ^ rotr_32(x, 11), 7) ^ (x >> 3))
|
||||||
|
#define s1(x) (rotr_32(x ^ rotr_32(x, 2), 17) ^ (x >> 10))
|
||||||
|
|
||||||
|
static const uint32_t SHA256_K[64] = {
|
||||||
|
0x428A2F98, 0x71374491, 0xB5C0FBCF, 0xE9B5DBA5,
|
||||||
|
0x3956C25B, 0x59F111F1, 0x923F82A4, 0xAB1C5ED5,
|
||||||
|
0xD807AA98, 0x12835B01, 0x243185BE, 0x550C7DC3,
|
||||||
|
0x72BE5D74, 0x80DEB1FE, 0x9BDC06A7, 0xC19BF174,
|
||||||
|
0xE49B69C1, 0xEFBE4786, 0x0FC19DC6, 0x240CA1CC,
|
||||||
|
0x2DE92C6F, 0x4A7484AA, 0x5CB0A9DC, 0x76F988DA,
|
||||||
|
0x983E5152, 0xA831C66D, 0xB00327C8, 0xBF597FC7,
|
||||||
|
0xC6E00BF3, 0xD5A79147, 0x06CA6351, 0x14292967,
|
||||||
|
0x27B70A85, 0x2E1B2138, 0x4D2C6DFC, 0x53380D13,
|
||||||
|
0x650A7354, 0x766A0ABB, 0x81C2C92E, 0x92722C85,
|
||||||
|
0xA2BFE8A1, 0xA81A664B, 0xC24B8B70, 0xC76C51A3,
|
||||||
|
0xD192E819, 0xD6990624, 0xF40E3585, 0x106AA070,
|
||||||
|
0x19A4C116, 0x1E376C08, 0x2748774C, 0x34B0BCB5,
|
||||||
|
0x391C0CB3, 0x4ED8AA4A, 0x5B9CCA4F, 0x682E6FF3,
|
||||||
|
0x748F82EE, 0x78A5636F, 0x84C87814, 0x8CC70208,
|
||||||
|
0x90BEFFFA, 0xA4506CEB, 0xBEF9A3F7, 0xC67178F2
|
||||||
|
};
|
||||||
|
|
||||||
|
static void
|
||||||
|
transform(uint32_t state[8], const uint8_t data[64])
|
||||||
|
{
|
||||||
|
uint32_t W[16];
|
||||||
|
uint32_t T[8];
|
||||||
|
unsigned int j;
|
||||||
|
|
||||||
|
/* Copy state[] to working vars. */
|
||||||
|
memcpy(T, state, sizeof(T));
|
||||||
|
|
||||||
|
/* The first 16 operations unrolled */
|
||||||
|
R0( 0); R0( 1); R0( 2); R0( 3);
|
||||||
|
R0( 4); R0( 5); R0( 6); R0( 7);
|
||||||
|
R0( 8); R0( 9); R0(10); R0(11);
|
||||||
|
R0(12); R0(13); R0(14); R0(15);
|
||||||
|
|
||||||
|
/* The remaining 48 operations partially unrolled */
|
||||||
|
for (j = 16; j < 64; j += 16) {
|
||||||
|
R2( 0); R2( 1); R2( 2); R2( 3);
|
||||||
|
R2( 4); R2( 5); R2( 6); R2( 7);
|
||||||
|
R2( 8); R2( 9); R2(10); R2(11);
|
||||||
|
R2(12); R2(13); R2(14); R2(15);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Add the working vars back into state[]. */
|
||||||
|
state[0] += a(0);
|
||||||
|
state[1] += b(0);
|
||||||
|
state[2] += c(0);
|
||||||
|
state[3] += d(0);
|
||||||
|
state[4] += e(0);
|
||||||
|
state[5] += f(0);
|
||||||
|
state[6] += g(0);
|
||||||
|
state[7] += h(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
XZ_EXTERN void xz_sha256_reset(struct xz_sha256 *s)
|
||||||
|
{
|
||||||
|
static const uint32_t initial_state[8] = {
|
||||||
|
0x6A09E667, 0xBB67AE85, 0x3C6EF372, 0xA54FF53A,
|
||||||
|
0x510E527F, 0x9B05688C, 0x1F83D9AB, 0x5BE0CD19
|
||||||
|
};
|
||||||
|
|
||||||
|
memcpy(s->state, initial_state, sizeof(initial_state));
|
||||||
|
s->size = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
XZ_EXTERN void xz_sha256_update(const uint8_t *buf, size_t size,
|
||||||
|
struct xz_sha256 *s)
|
||||||
|
{
|
||||||
|
size_t copy_start;
|
||||||
|
size_t copy_size;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Copy the input data into a properly aligned temporary buffer.
|
||||||
|
* This way we can be called with arbitrarily sized buffers
|
||||||
|
* (no need to be a multiple of 64 bytes).
|
||||||
|
*
|
||||||
|
* Full 64-byte chunks could be processed directly from buf with
|
||||||
|
* unaligned access. It seemed to make very little difference in
|
||||||
|
* speed on x86-64 though. Thus it was omitted.
|
||||||
|
*/
|
||||||
|
while (size > 0) {
|
||||||
|
copy_start = s->size & 0x3F;
|
||||||
|
copy_size = 64 - copy_start;
|
||||||
|
if (copy_size > size)
|
||||||
|
copy_size = size;
|
||||||
|
|
||||||
|
memcpy(s->data + copy_start, buf, copy_size);
|
||||||
|
|
||||||
|
buf += copy_size;
|
||||||
|
size -= copy_size;
|
||||||
|
s->size += copy_size;
|
||||||
|
|
||||||
|
if ((s->size & 0x3F) == 0)
|
||||||
|
transform(s->state, s->data);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
XZ_EXTERN bool xz_sha256_validate(const uint8_t *buf, struct xz_sha256 *s)
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
* Add padding as described in RFC 3174 (it describes SHA-1 but
|
||||||
|
* the same padding style is used for SHA-256 too).
|
||||||
|
*/
|
||||||
|
size_t i = s->size & 0x3F;
|
||||||
|
s->data[i++] = 0x80;
|
||||||
|
|
||||||
|
while (i != 64 - 8) {
|
||||||
|
if (i == 64) {
|
||||||
|
transform(s->state, s->data);
|
||||||
|
i = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
s->data[i++] = 0x00;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Convert the message size from bytes to bits. */
|
||||||
|
s->size *= 8;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Store the message size in big endian byte order and
|
||||||
|
* calculate the final hash value.
|
||||||
|
*/
|
||||||
|
for (i = 0; i < 8; ++i)
|
||||||
|
s->data[64 - 8 + i] = (uint8_t)(s->size >> ((7 - i) * 8));
|
||||||
|
|
||||||
|
transform(s->state, s->data);
|
||||||
|
|
||||||
|
/* Compare if the hash value matches the first 32 bytes in buf. */
|
||||||
|
for (i = 0; i < 8; ++i)
|
||||||
|
if (get_unaligned_be32(buf + 4 * i) != s->state[i])
|
||||||
|
return false;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
61
android/app/src/main/cpp/xz/xz_stream.h
Normal file
61
android/app/src/main/cpp/xz/xz_stream.h
Normal file
@@ -0,0 +1,61 @@
|
|||||||
|
/* SPDX-License-Identifier: 0BSD */
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Definitions for handling the .xz file format
|
||||||
|
*
|
||||||
|
* Author: Lasse Collin <lasse.collin@tukaani.org>
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef XZ_STREAM_H
|
||||||
|
#define XZ_STREAM_H
|
||||||
|
|
||||||
|
#if defined(__KERNEL__) && !XZ_INTERNAL_CRC32
|
||||||
|
# include <linux/crc32.h>
|
||||||
|
# undef crc32
|
||||||
|
# define xz_crc32(buf, size, crc) \
|
||||||
|
(~crc32_le(~(uint32_t)(crc), buf, size))
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/*
|
||||||
|
* See the .xz file format specification at
|
||||||
|
* https://tukaani.org/xz/xz-file-format.txt
|
||||||
|
* to understand the container format.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#define STREAM_HEADER_SIZE 12
|
||||||
|
|
||||||
|
#define HEADER_MAGIC "\3757zXZ"
|
||||||
|
#define HEADER_MAGIC_SIZE 6
|
||||||
|
|
||||||
|
#define FOOTER_MAGIC "YZ"
|
||||||
|
#define FOOTER_MAGIC_SIZE 2
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Variable-length integer can hold a 63-bit unsigned integer or a special
|
||||||
|
* value indicating that the value is unknown.
|
||||||
|
*
|
||||||
|
* Experimental: vli_type can be defined to uint32_t to save a few bytes
|
||||||
|
* in code size (no effect on speed). Doing so limits the uncompressed and
|
||||||
|
* compressed size of the file to less than 256 MiB and may also weaken
|
||||||
|
* error detection slightly.
|
||||||
|
*/
|
||||||
|
typedef uint64_t vli_type;
|
||||||
|
|
||||||
|
#define VLI_MAX ((vli_type)-1 / 2)
|
||||||
|
#define VLI_UNKNOWN ((vli_type)-1)
|
||||||
|
|
||||||
|
/* Maximum encoded size of a VLI */
|
||||||
|
#define VLI_BYTES_MAX (sizeof(vli_type) * 8 / 7)
|
||||||
|
|
||||||
|
/* Integrity Check types */
|
||||||
|
enum xz_check {
|
||||||
|
XZ_CHECK_NONE = 0,
|
||||||
|
XZ_CHECK_CRC32 = 1,
|
||||||
|
XZ_CHECK_CRC64 = 4,
|
||||||
|
XZ_CHECK_SHA256 = 10
|
||||||
|
};
|
||||||
|
|
||||||
|
/* Maximum possible Check ID */
|
||||||
|
#define XZ_CHECK_MAX 15
|
||||||
|
|
||||||
|
#endif
|
||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@file:OptIn(ExperimentalEncodingApi::class)
|
@file:OptIn(ExperimentalEncodingApi::class)
|
||||||
|
|
||||||
@@ -126,7 +126,7 @@ import me.kavishdevar.librepods.screens.HearingAidAdjustmentsScreen
|
|||||||
import me.kavishdevar.librepods.screens.HearingAidScreen
|
import me.kavishdevar.librepods.screens.HearingAidScreen
|
||||||
import me.kavishdevar.librepods.screens.HearingProtectionScreen
|
import me.kavishdevar.librepods.screens.HearingProtectionScreen
|
||||||
import me.kavishdevar.librepods.screens.LongPress
|
import me.kavishdevar.librepods.screens.LongPress
|
||||||
import me.kavishdevar.librepods.screens.Onboarding
|
// import me.kavishdevar.librepods.screens.Onboarding
|
||||||
import me.kavishdevar.librepods.screens.OpenSourceLicensesScreen
|
import me.kavishdevar.librepods.screens.OpenSourceLicensesScreen
|
||||||
import me.kavishdevar.librepods.screens.RenameScreen
|
import me.kavishdevar.librepods.screens.RenameScreen
|
||||||
import me.kavishdevar.librepods.screens.TransparencySettingsScreen
|
import me.kavishdevar.librepods.screens.TransparencySettingsScreen
|
||||||
@@ -135,7 +135,7 @@ import me.kavishdevar.librepods.screens.UpdateHearingTestScreen
|
|||||||
import me.kavishdevar.librepods.screens.VersionScreen
|
import me.kavishdevar.librepods.screens.VersionScreen
|
||||||
import me.kavishdevar.librepods.services.AirPodsService
|
import me.kavishdevar.librepods.services.AirPodsService
|
||||||
import me.kavishdevar.librepods.ui.theme.LibrePodsTheme
|
import me.kavishdevar.librepods.ui.theme.LibrePodsTheme
|
||||||
import me.kavishdevar.librepods.utils.RadareOffsetFinder
|
// import me.kavishdevar.librepods.utils.RadareOffsetFinder
|
||||||
import kotlin.io.encoding.Base64
|
import kotlin.io.encoding.Base64
|
||||||
import kotlin.io.encoding.ExperimentalEncodingApi
|
import kotlin.io.encoding.ExperimentalEncodingApi
|
||||||
|
|
||||||
@@ -156,10 +156,6 @@ class MainActivity : ComponentActivity() {
|
|||||||
|
|
||||||
setContent {
|
setContent {
|
||||||
LibrePodsTheme {
|
LibrePodsTheme {
|
||||||
getSharedPreferences("settings", MODE_PRIVATE).edit {
|
|
||||||
putLong(
|
|
||||||
"textColor",
|
|
||||||
MaterialTheme.colorScheme.onSurface.toArgb().toLong())}
|
|
||||||
Main()
|
Main()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -271,7 +267,7 @@ class MainActivity : ComponentActivity() {
|
|||||||
fun Main() {
|
fun Main() {
|
||||||
val isConnected = remember { mutableStateOf(false) }
|
val isConnected = remember { mutableStateOf(false) }
|
||||||
val isRemotelyConnected = remember { mutableStateOf(false) }
|
val isRemotelyConnected = remember { mutableStateOf(false) }
|
||||||
val hookAvailable = RadareOffsetFinder(LocalContext.current).isHookOffsetAvailable()
|
// val hookAvailable = RadareOffsetFinder(LocalContext.current).isHookOffsetAvailable()
|
||||||
val context = LocalContext.current
|
val context = LocalContext.current
|
||||||
var canDrawOverlays by remember { mutableStateOf(Settings.canDrawOverlays(context)) }
|
var canDrawOverlays by remember { mutableStateOf(Settings.canDrawOverlays(context)) }
|
||||||
val overlaySkipped = remember { mutableStateOf(context.getSharedPreferences("settings", MODE_PRIVATE).getBoolean("overlay_permission_skipped", false)) }
|
val overlaySkipped = remember { mutableStateOf(context.getSharedPreferences("settings", MODE_PRIVATE).getBoolean("overlay_permission_skipped", false)) }
|
||||||
@@ -325,7 +321,7 @@ fun Main() {
|
|||||||
) {
|
) {
|
||||||
NavHost(
|
NavHost(
|
||||||
navController = navController,
|
navController = navController,
|
||||||
startDestination = if (hookAvailable) "settings" else "onboarding",
|
startDestination = "settings", // if (hookAvailable) "settings" else "onboarding",
|
||||||
enterTransition = {
|
enterTransition = {
|
||||||
slideInHorizontally(
|
slideInHorizontally(
|
||||||
initialOffsetX = { it },
|
initialOffsetX = { it },
|
||||||
@@ -381,11 +377,11 @@ fun Main() {
|
|||||||
TroubleshootingScreen(navController)
|
TroubleshootingScreen(navController)
|
||||||
}
|
}
|
||||||
composable("head_tracking") {
|
composable("head_tracking") {
|
||||||
HeadTrackingScreen(navController)
|
HeadTrackingScreen()
|
||||||
}
|
}
|
||||||
composable("onboarding") {
|
/*composable("onboarding") {
|
||||||
Onboarding(navController, context)
|
Onboarding(navController, context)
|
||||||
}
|
}*/
|
||||||
composable("accessibility") {
|
composable("accessibility") {
|
||||||
AccessibilitySettingsScreen(navController)
|
AccessibilitySettingsScreen(navController)
|
||||||
}
|
}
|
||||||
@@ -423,7 +419,7 @@ fun Main() {
|
|||||||
|
|
||||||
LaunchedEffect(navController) {
|
LaunchedEffect(navController) {
|
||||||
navController.addOnDestinationChangedListener { _, destination, _ ->
|
navController.addOnDestinationChangedListener { _, destination, _ ->
|
||||||
showBackButton.value = destination.route != "settings" && destination.route != "onboarding"
|
showBackButton.value = destination.route != "settings" // && destination.route != "onboarding"
|
||||||
Log.d("MainActivity", "Navigated to ${destination.route}, showBackButton: ${showBackButton.value}")
|
Log.d("MainActivity", "Navigated to ${destination.route}, showBackButton: ${showBackButton.value}")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@file:OptIn(ExperimentalEncodingApi::class)
|
@file:OptIn(ExperimentalEncodingApi::class)
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@file:OptIn(ExperimentalEncodingApi::class)
|
@file:OptIn(ExperimentalEncodingApi::class)
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@file:OptIn(ExperimentalEncodingApi::class)
|
@file:OptIn(ExperimentalEncodingApi::class)
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.composables
|
package me.kavishdevar.librepods.composables
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@file:OptIn(ExperimentalEncodingApi::class)
|
@file:OptIn(ExperimentalEncodingApi::class)
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@file:OptIn(ExperimentalEncodingApi::class)
|
@file:OptIn(ExperimentalEncodingApi::class)
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.composables
|
package me.kavishdevar.librepods.composables
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@file:OptIn(ExperimentalEncodingApi::class)
|
@file:OptIn(ExperimentalEncodingApi::class)
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods Contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@file:Suppress("unused")
|
@file:Suppress("unused")
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods Contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.composables
|
package me.kavishdevar.librepods.composables
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@file:OptIn(ExperimentalEncodingApi::class)
|
@file:OptIn(ExperimentalEncodingApi::class)
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@file:OptIn(ExperimentalEncodingApi::class)
|
@file:OptIn(ExperimentalEncodingApi::class)
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.composables
|
package me.kavishdevar.librepods.composables
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.composables
|
package me.kavishdevar.librepods.composables
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@file:OptIn(ExperimentalEncodingApi::class)
|
@file:OptIn(ExperimentalEncodingApi::class)
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.composables
|
package me.kavishdevar.librepods.composables
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.composables
|
package me.kavishdevar.librepods.composables
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.composables
|
package me.kavishdevar.librepods.composables
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.composables
|
package me.kavishdevar.librepods.composables
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.composables
|
package me.kavishdevar.librepods.composables
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.composables
|
package me.kavishdevar.librepods.composables
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.composables
|
package me.kavishdevar.librepods.composables
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.composables
|
package me.kavishdevar.librepods.composables
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@file:OptIn(ExperimentalEncodingApi::class)
|
@file:OptIn(ExperimentalEncodingApi::class)
|
||||||
|
|
||||||
@@ -472,7 +472,12 @@ fun StyledToggle(
|
|||||||
val attManager = ServiceManager.getService()?.attManager ?: return
|
val attManager = ServiceManager.getService()?.attManager ?: return
|
||||||
val isDarkTheme = isSystemInDarkTheme()
|
val isDarkTheme = isSystemInDarkTheme()
|
||||||
val textColor = if (isDarkTheme) Color.White else Color.Black
|
val textColor = if (isDarkTheme) Color.White else Color.Black
|
||||||
val checkedValue = attManager.read(attHandle).getOrNull(0)?.toInt()
|
val checkedValue = try {
|
||||||
|
attManager.read(attHandle).getOrNull(0)?.toInt()
|
||||||
|
} catch (e: Exception) {
|
||||||
|
Log.w("StyledToggle", "Error reading initial value for $label: ${e.message}")
|
||||||
|
null
|
||||||
|
} ?: 0
|
||||||
var checked by remember { mutableStateOf(checkedValue !=0) }
|
var checked by remember { mutableStateOf(checkedValue !=0) }
|
||||||
var backgroundColor by remember { mutableStateOf(if (isDarkTheme) Color(0xFF1C1C1E) else Color(0xFFFFFFFF)) }
|
var backgroundColor by remember { mutableStateOf(if (isDarkTheme) Color(0xFF1C1C1E) else Color(0xFFFFFFFF)) }
|
||||||
val animatedBackgroundColor by animateColorAsState(targetValue = backgroundColor, animationSpec = tween(durationMillis = 500))
|
val animatedBackgroundColor by animateColorAsState(targetValue = backgroundColor, animationSpec = tween(durationMillis = 500))
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods Contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.composables
|
package me.kavishdevar.librepods.composables
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.constants
|
package me.kavishdevar.librepods.constants
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.constants
|
package me.kavishdevar.librepods.constants
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@file:OptIn(ExperimentalEncodingApi::class)
|
@file:OptIn(ExperimentalEncodingApi::class)
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.screens
|
package me.kavishdevar.librepods.screens
|
||||||
|
|
||||||
@@ -84,7 +84,7 @@ import me.kavishdevar.librepods.services.ServiceManager
|
|||||||
import me.kavishdevar.librepods.utils.AACPManager
|
import me.kavishdevar.librepods.utils.AACPManager
|
||||||
import me.kavishdevar.librepods.utils.ATTHandles
|
import me.kavishdevar.librepods.utils.ATTHandles
|
||||||
import me.kavishdevar.librepods.utils.Capability
|
import me.kavishdevar.librepods.utils.Capability
|
||||||
import me.kavishdevar.librepods.utils.RadareOffsetFinder
|
// import me.kavishdevar.librepods.utils.RadareOffsetFinder
|
||||||
import kotlin.io.encoding.ExperimentalEncodingApi
|
import kotlin.io.encoding.ExperimentalEncodingApi
|
||||||
|
|
||||||
private var phoneMediaDebounceJob: Job? = null
|
private var phoneMediaDebounceJob: Job? = null
|
||||||
@@ -99,8 +99,8 @@ fun AccessibilitySettingsScreen(navController: NavController) {
|
|||||||
val isDarkTheme = isSystemInDarkTheme()
|
val isDarkTheme = isSystemInDarkTheme()
|
||||||
val textColor = if (isDarkTheme) Color.White else Color.Black
|
val textColor = if (isDarkTheme) Color.White else Color.Black
|
||||||
val aacpManager = remember { ServiceManager.getService()?.aacpManager }
|
val aacpManager = remember { ServiceManager.getService()?.aacpManager }
|
||||||
val isSdpOffsetAvailable =
|
val isSdpOffsetAvailable = remember { mutableStateOf(false) } // always available rn, for testing without radare
|
||||||
remember { mutableStateOf(RadareOffsetFinder.isSdpOffsetAvailable()) }
|
// remember { mutableStateOf(RadareOffsetFinder.isSdpOffsetAvailable()) }
|
||||||
|
|
||||||
val trackColor = if (isDarkTheme) Color(0xFFB3B3B3) else Color(0xFF929491)
|
val trackColor = if (isDarkTheme) Color(0xFFB3B3B3) else Color(0xFF929491)
|
||||||
val activeTrackColor = if (isDarkTheme) Color(0xFF007AFF) else Color(0xFF3C6DF5)
|
val activeTrackColor = if (isDarkTheme) Color(0xFF007AFF) else Color(0xFF3C6DF5)
|
||||||
@@ -160,9 +160,9 @@ fun AccessibilitySettingsScreen(navController: NavController) {
|
|||||||
val mediaEQEnabled = remember { mutableStateOf(false) }
|
val mediaEQEnabled = remember { mutableStateOf(false) }
|
||||||
|
|
||||||
val pressSpeedOptions = mapOf(
|
val pressSpeedOptions = mapOf(
|
||||||
0.toByte() to "Default",
|
0.toByte() to stringResource(R.string.default_option),
|
||||||
1.toByte() to "Slower",
|
1.toByte() to stringResource(R.string.slower),
|
||||||
2.toByte() to "Slowest"
|
2.toByte() to stringResource(R.string.slowest)
|
||||||
)
|
)
|
||||||
val selectedPressSpeedValue =
|
val selectedPressSpeedValue =
|
||||||
aacpManager?.controlCommandStatusList?.find { it.identifier == AACPManager.Companion.ControlCommandIdentifiers.DOUBLE_CLICK_INTERVAL }?.value?.takeIf { it.isNotEmpty() }
|
aacpManager?.controlCommandStatusList?.find { it.identifier == AACPManager.Companion.ControlCommandIdentifiers.DOUBLE_CLICK_INTERVAL }?.value?.takeIf { it.isNotEmpty() }
|
||||||
@@ -196,9 +196,9 @@ fun AccessibilitySettingsScreen(navController: NavController) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
val pressAndHoldDurationOptions = mapOf(
|
val pressAndHoldDurationOptions = mapOf(
|
||||||
0.toByte() to "Default",
|
0.toByte() to stringResource(R.string.default_option),
|
||||||
1.toByte() to "Slower",
|
1.toByte() to stringResource(R.string.slower),
|
||||||
2.toByte() to "Slowest"
|
2.toByte() to stringResource(R.string.slowest)
|
||||||
)
|
)
|
||||||
val selectedPressAndHoldDurationValue =
|
val selectedPressAndHoldDurationValue =
|
||||||
aacpManager?.controlCommandStatusList?.find { it.identifier == AACPManager.Companion.ControlCommandIdentifiers.CLICK_HOLD_INTERVAL }?.value?.takeIf { it.isNotEmpty() }
|
aacpManager?.controlCommandStatusList?.find { it.identifier == AACPManager.Companion.ControlCommandIdentifiers.CLICK_HOLD_INTERVAL }?.value?.takeIf { it.isNotEmpty() }
|
||||||
@@ -234,9 +234,9 @@ fun AccessibilitySettingsScreen(navController: NavController) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
val volumeSwipeSpeedOptions = mapOf(
|
val volumeSwipeSpeedOptions = mapOf(
|
||||||
1.toByte() to "Default",
|
1.toByte() to stringResource(R.string.default_option),
|
||||||
2.toByte() to "Longer",
|
2.toByte() to stringResource(R.string.longer),
|
||||||
3.toByte() to "Longest"
|
3.toByte() to stringResource(R.string.longest)
|
||||||
)
|
)
|
||||||
val selectedVolumeSwipeSpeedValue =
|
val selectedVolumeSwipeSpeedValue =
|
||||||
aacpManager?.controlCommandStatusList?.find { it.identifier == AACPManager.Companion.ControlCommandIdentifiers.VOLUME_SWIPE_INTERVAL }?.value?.takeIf { it.isNotEmpty() }
|
aacpManager?.controlCommandStatusList?.find { it.identifier == AACPManager.Companion.ControlCommandIdentifiers.VOLUME_SWIPE_INTERVAL }?.value?.takeIf { it.isNotEmpty() }
|
||||||
@@ -322,7 +322,7 @@ fun AccessibilitySettingsScreen(navController: NavController) {
|
|||||||
label = stringResource(R.string.press_speed),
|
label = stringResource(R.string.press_speed),
|
||||||
description = stringResource(R.string.press_speed_description),
|
description = stringResource(R.string.press_speed_description),
|
||||||
options = pressSpeedOptions.values.toList(),
|
options = pressSpeedOptions.values.toList(),
|
||||||
selectedOption = selectedPressSpeed?: "Default",
|
selectedOption = selectedPressSpeed?: stringResource(R.string.default_option),
|
||||||
onOptionSelected = { newValue ->
|
onOptionSelected = { newValue ->
|
||||||
selectedPressSpeed = newValue
|
selectedPressSpeed = newValue
|
||||||
aacpManager?.sendControlCommand(
|
aacpManager?.sendControlCommand(
|
||||||
@@ -340,7 +340,7 @@ fun AccessibilitySettingsScreen(navController: NavController) {
|
|||||||
label = stringResource(R.string.press_and_hold_duration),
|
label = stringResource(R.string.press_and_hold_duration),
|
||||||
description = stringResource(R.string.press_and_hold_duration_description),
|
description = stringResource(R.string.press_and_hold_duration_description),
|
||||||
options = pressAndHoldDurationOptions.values.toList(),
|
options = pressAndHoldDurationOptions.values.toList(),
|
||||||
selectedOption = selectedPressAndHoldDuration?: "Default",
|
selectedOption = selectedPressAndHoldDuration?: stringResource(R.string.default_option),
|
||||||
onOptionSelected = { newValue ->
|
onOptionSelected = { newValue ->
|
||||||
selectedPressAndHoldDuration = newValue
|
selectedPressAndHoldDuration = newValue
|
||||||
aacpManager?.sendControlCommand(
|
aacpManager?.sendControlCommand(
|
||||||
@@ -403,7 +403,7 @@ fun AccessibilitySettingsScreen(navController: NavController) {
|
|||||||
label = stringResource(R.string.volume_swipe_speed),
|
label = stringResource(R.string.volume_swipe_speed),
|
||||||
description = stringResource(R.string.volume_swipe_speed_description),
|
description = stringResource(R.string.volume_swipe_speed_description),
|
||||||
options = volumeSwipeSpeedOptions.values.toList(),
|
options = volumeSwipeSpeedOptions.values.toList(),
|
||||||
selectedOption = selectedVolumeSwipeSpeed?: "Default",
|
selectedOption = selectedVolumeSwipeSpeed?: stringResource(R.string.default_option),
|
||||||
onOptionSelected = { newValue ->
|
onOptionSelected = { newValue ->
|
||||||
selectedVolumeSwipeSpeed = newValue
|
selectedVolumeSwipeSpeed = newValue
|
||||||
aacpManager?.sendControlCommand(
|
aacpManager?.sendControlCommand(
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.screens
|
package me.kavishdevar.librepods.screens
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@file:OptIn(ExperimentalEncodingApi::class)
|
@file:OptIn(ExperimentalEncodingApi::class)
|
||||||
|
|
||||||
@@ -97,7 +97,7 @@ import me.kavishdevar.librepods.services.AirPodsService
|
|||||||
import me.kavishdevar.librepods.ui.theme.LibrePodsTheme
|
import me.kavishdevar.librepods.ui.theme.LibrePodsTheme
|
||||||
import me.kavishdevar.librepods.utils.AACPManager
|
import me.kavishdevar.librepods.utils.AACPManager
|
||||||
import me.kavishdevar.librepods.utils.Capability
|
import me.kavishdevar.librepods.utils.Capability
|
||||||
import me.kavishdevar.librepods.utils.RadareOffsetFinder
|
// import me.kavishdevar.librepods.utils.RadareOffsetFinder
|
||||||
import kotlin.io.encoding.ExperimentalEncodingApi
|
import kotlin.io.encoding.ExperimentalEncodingApi
|
||||||
|
|
||||||
@OptIn(ExperimentalMaterial3Api::class, ExperimentalHazeMaterialsApi::class)
|
@OptIn(ExperimentalMaterial3Api::class, ExperimentalHazeMaterialsApi::class)
|
||||||
@@ -218,7 +218,9 @@ fun AirPodsSettingsScreen(dev: BluetoothDevice?, service: AirPodsService,
|
|||||||
val darkMode = isSystemInDarkTheme()
|
val darkMode = isSystemInDarkTheme()
|
||||||
val hazeStateS = remember { mutableStateOf(HazeState()) }
|
val hazeStateS = remember { mutableStateOf(HazeState()) }
|
||||||
|
|
||||||
val showDialog = remember { mutableStateOf(!sharedPreferences.getBoolean("donationDialogShown", false)) }
|
// val showDialog = remember { mutableStateOf(!sharedPreferences.getBoolean("donationDialogShown", false)) }
|
||||||
|
|
||||||
|
val showDialog = remember { mutableStateOf(false) }
|
||||||
|
|
||||||
StyledScaffold(
|
StyledScaffold(
|
||||||
title = deviceName.text,
|
title = deviceName.text,
|
||||||
@@ -263,13 +265,13 @@ fun AirPodsSettingsScreen(dev: BluetoothDevice?, service: AirPodsService,
|
|||||||
independent = true
|
independent = true
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
val actAsAppleDeviceHookEnabled = RadareOffsetFinder.isSdpOffsetAvailable()
|
// val actAsAppleDeviceHookEnabled = RadareOffsetFinder.isSdpOffsetAvailable()
|
||||||
if (actAsAppleDeviceHookEnabled) {
|
// if (actAsAppleDeviceHookEnabled) {
|
||||||
item(key = "spacer_hearing_health") { Spacer(modifier = Modifier.height(32.dp)) }
|
// item(key = "spacer_hearing_health") { Spacer(modifier = Modifier.height(32.dp)) }
|
||||||
item(key = "hearing_health") {
|
// item(key = "hearing_health") {
|
||||||
HearingHealthSettings(navController = navController)
|
// HearingHealthSettings(navController = navController)
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
|
|
||||||
if (capabilities.contains(Capability.LISTENING_MODE)) {
|
if (capabilities.contains(Capability.LISTENING_MODE)) {
|
||||||
item(key = "spacer_noise") { Spacer(modifier = Modifier.height(16.dp)) }
|
item(key = "spacer_noise") { Spacer(modifier = Modifier.height(16.dp)) }
|
||||||
@@ -384,7 +386,7 @@ fun AirPodsSettingsScreen(dev: BluetoothDevice?, service: AirPodsService,
|
|||||||
.fillMaxWidth(0.9f)
|
.fillMaxWidth(0.9f)
|
||||||
) {
|
) {
|
||||||
Text(
|
Text(
|
||||||
text = "Troubleshoot Connection",
|
text = stringResource(R.string.troubleshooting),
|
||||||
style = TextStyle(
|
style = TextStyle(
|
||||||
fontSize = 16.sp,
|
fontSize = 16.sp,
|
||||||
fontWeight = FontWeight.Medium,
|
fontWeight = FontWeight.Medium,
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.screens
|
package me.kavishdevar.librepods.screens
|
||||||
|
|
||||||
@@ -84,7 +84,7 @@ import me.kavishdevar.librepods.composables.StyledScaffold
|
|||||||
import me.kavishdevar.librepods.composables.StyledSlider
|
import me.kavishdevar.librepods.composables.StyledSlider
|
||||||
import me.kavishdevar.librepods.composables.StyledToggle
|
import me.kavishdevar.librepods.composables.StyledToggle
|
||||||
import me.kavishdevar.librepods.utils.AACPManager
|
import me.kavishdevar.librepods.utils.AACPManager
|
||||||
import me.kavishdevar.librepods.utils.RadareOffsetFinder
|
//import me.kavishdevar.librepods.utils.RadareOffsetFinder
|
||||||
import kotlin.io.encoding.Base64
|
import kotlin.io.encoding.Base64
|
||||||
import kotlin.io.encoding.ExperimentalEncodingApi
|
import kotlin.io.encoding.ExperimentalEncodingApi
|
||||||
import kotlin.math.roundToInt
|
import kotlin.math.roundToInt
|
||||||
@@ -185,7 +185,7 @@ fun AppSettingsScreen(navController: NavController) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
val isProcessingSdp = remember { mutableStateOf(false) }
|
val isProcessingSdp = remember { mutableStateOf(false) }
|
||||||
val actAsAppleDevice = remember { mutableStateOf(false) }
|
// val actAsAppleDevice = remember { mutableStateOf(false) }
|
||||||
|
|
||||||
BackHandler(enabled = isProcessingSdp.value) {}
|
BackHandler(enabled = isProcessingSdp.value) {}
|
||||||
|
|
||||||
@@ -575,12 +575,12 @@ fun AppSettingsScreen(navController: NavController) {
|
|||||||
description = stringResource(R.string.troubleshooting_description)
|
description = stringResource(R.string.troubleshooting_description)
|
||||||
)
|
)
|
||||||
|
|
||||||
LaunchedEffect(Unit) {
|
// LaunchedEffect(Unit) {
|
||||||
actAsAppleDevice.value = RadareOffsetFinder.isSdpOffsetAvailable()
|
// actAsAppleDevice.value = RadareOffsetFinder.isSdpOffsetAvailable()
|
||||||
}
|
// }
|
||||||
val restartBluetoothText = stringResource(R.string.found_offset_restart_bluetooth)
|
val restartBluetoothText = stringResource(R.string.found_offset_restart_bluetooth)
|
||||||
|
|
||||||
StyledToggle(
|
/*StyledToggle(
|
||||||
label = stringResource(R.string.act_as_an_apple_device),
|
label = stringResource(R.string.act_as_an_apple_device),
|
||||||
description = stringResource(R.string.act_as_an_apple_device_description),
|
description = stringResource(R.string.act_as_an_apple_device_description),
|
||||||
checkedState = actAsAppleDevice,
|
checkedState = actAsAppleDevice,
|
||||||
@@ -602,7 +602,7 @@ fun AppSettingsScreen(navController: NavController) {
|
|||||||
},
|
},
|
||||||
independent = true,
|
independent = true,
|
||||||
enabled = !isProcessingSdp.value
|
enabled = !isProcessingSdp.value
|
||||||
)
|
)*/
|
||||||
|
|
||||||
Spacer(modifier = Modifier.height(16.dp))
|
Spacer(modifier = Modifier.height(16.dp))
|
||||||
|
|
||||||
@@ -650,70 +650,70 @@ fun AppSettingsScreen(navController: NavController) {
|
|||||||
|
|
||||||
Spacer(modifier = Modifier.height(32.dp))
|
Spacer(modifier = Modifier.height(32.dp))
|
||||||
|
|
||||||
if (showResetDialog.value) {
|
// if (showResetDialog.value) {
|
||||||
AlertDialog(
|
// AlertDialog(
|
||||||
onDismissRequest = { showResetDialog.value = false },
|
// onDismissRequest = { showResetDialog.value = false },
|
||||||
title = {
|
// title = {
|
||||||
Text(
|
// Text(
|
||||||
"Reset Hook Offset",
|
// "Reset Hook Offset",
|
||||||
fontFamily = FontFamily(Font(R.font.sf_pro)),
|
// fontFamily = FontFamily(Font(R.font.sf_pro)),
|
||||||
fontWeight = FontWeight.Medium
|
// fontWeight = FontWeight.Medium
|
||||||
)
|
// )
|
||||||
},
|
// },
|
||||||
text = {
|
// text = {
|
||||||
Text(
|
// Text(
|
||||||
stringResource(R.string.reset_hook_offset_description),
|
// stringResource(R.string.reset_hook_offset_description),
|
||||||
fontFamily = FontFamily(Font(R.font.sf_pro))
|
// fontFamily = FontFamily(Font(R.font.sf_pro))
|
||||||
)
|
// )
|
||||||
},
|
// },
|
||||||
confirmButton = {
|
// confirmButton = {
|
||||||
val successText = stringResource(R.string.hook_offset_reset_success)
|
// val successText = stringResource(R.string.hook_offset_reset_success)
|
||||||
val failureText = stringResource(R.string.hook_offset_reset_failure)
|
// val failureText = stringResource(R.string.hook_offset_reset_failure)
|
||||||
TextButton(
|
// TextButton(
|
||||||
onClick = {
|
// onClick = {
|
||||||
if (RadareOffsetFinder.clearHookOffsets()) {
|
// if (RadareOffsetFinder.clearHookOffsets()) {
|
||||||
Toast.makeText(
|
// Toast.makeText(
|
||||||
context,
|
// context,
|
||||||
successText,
|
// successText,
|
||||||
Toast.LENGTH_LONG
|
// Toast.LENGTH_LONG
|
||||||
).show()
|
// ).show()
|
||||||
|
//
|
||||||
navController.navigate("onboarding") {
|
// navController.navigate("onboarding") {
|
||||||
popUpTo("settings") { inclusive = true }
|
// popUpTo("settings") { inclusive = true }
|
||||||
}
|
// }
|
||||||
} else {
|
// } else {
|
||||||
Toast.makeText(
|
// Toast.makeText(
|
||||||
context,
|
// context,
|
||||||
failureText,
|
// failureText,
|
||||||
Toast.LENGTH_SHORT
|
// Toast.LENGTH_SHORT
|
||||||
).show()
|
// ).show()
|
||||||
}
|
// }
|
||||||
showResetDialog.value = false
|
// showResetDialog.value = false
|
||||||
},
|
// },
|
||||||
colors = ButtonDefaults.textButtonColors(
|
// colors = ButtonDefaults.textButtonColors(
|
||||||
contentColor = MaterialTheme.colorScheme.error
|
// contentColor = MaterialTheme.colorScheme.error
|
||||||
)
|
// )
|
||||||
) {
|
// ) {
|
||||||
Text(
|
// Text(
|
||||||
stringResource(R.string.reset),
|
// stringResource(R.string.reset),
|
||||||
fontFamily = FontFamily(Font(R.font.sf_pro)),
|
// fontFamily = FontFamily(Font(R.font.sf_pro)),
|
||||||
fontWeight = FontWeight.Medium
|
// fontWeight = FontWeight.Medium
|
||||||
)
|
// )
|
||||||
}
|
// }
|
||||||
},
|
// },
|
||||||
dismissButton = {
|
// dismissButton = {
|
||||||
TextButton(
|
// TextButton(
|
||||||
onClick = { showResetDialog.value = false }
|
// onClick = { showResetDialog.value = false }
|
||||||
) {
|
// ) {
|
||||||
Text(
|
// Text(
|
||||||
"Cancel",
|
// "Cancel",
|
||||||
fontFamily = FontFamily(Font(R.font.sf_pro)),
|
// fontFamily = FontFamily(Font(R.font.sf_pro)),
|
||||||
fontWeight = FontWeight.Medium
|
// fontWeight = FontWeight.Medium
|
||||||
)
|
// )
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
)
|
// )
|
||||||
}
|
// }
|
||||||
|
|
||||||
if (showIrkDialog.value) {
|
if (showIrkDialog.value) {
|
||||||
AlertDialog(
|
AlertDialog(
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.screens
|
package me.kavishdevar.librepods.screens
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@file:OptIn(ExperimentalHazeMaterialsApi::class, ExperimentalEncodingApi::class)
|
@file:OptIn(ExperimentalHazeMaterialsApi::class, ExperimentalEncodingApi::class)
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,23 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
// this is absolutely unnecessary, why did I make this. a simple toggle would've sufficed
|
||||||
|
|
||||||
@file:OptIn(ExperimentalEncodingApi::class)
|
@file:OptIn(ExperimentalEncodingApi::class)
|
||||||
|
|
||||||
@@ -83,7 +86,6 @@ import androidx.compose.ui.text.style.TextAlign
|
|||||||
import androidx.compose.ui.tooling.preview.Preview
|
import androidx.compose.ui.tooling.preview.Preview
|
||||||
import androidx.compose.ui.unit.dp
|
import androidx.compose.ui.unit.dp
|
||||||
import androidx.compose.ui.unit.sp
|
import androidx.compose.ui.unit.sp
|
||||||
import androidx.navigation.NavController
|
|
||||||
import com.kyant.backdrop.backdrops.layerBackdrop
|
import com.kyant.backdrop.backdrops.layerBackdrop
|
||||||
import com.kyant.backdrop.backdrops.rememberLayerBackdrop
|
import com.kyant.backdrop.backdrops.rememberLayerBackdrop
|
||||||
import dev.chrisbanes.haze.hazeSource
|
import dev.chrisbanes.haze.hazeSource
|
||||||
@@ -108,7 +110,7 @@ import kotlin.random.Random
|
|||||||
@RequiresApi(Build.VERSION_CODES.Q)
|
@RequiresApi(Build.VERSION_CODES.Q)
|
||||||
@OptIn(ExperimentalMaterial3Api::class, ExperimentalAnimationApi::class)
|
@OptIn(ExperimentalMaterial3Api::class, ExperimentalAnimationApi::class)
|
||||||
@Composable
|
@Composable
|
||||||
fun HeadTrackingScreen(navController: NavController) {
|
fun HeadTrackingScreen() {
|
||||||
DisposableEffect(Unit) {
|
DisposableEffect(Unit) {
|
||||||
ServiceManager.getService()?.startHeadTracking()
|
ServiceManager.getService()?.startHeadTracking()
|
||||||
onDispose {
|
onDispose {
|
||||||
@@ -743,5 +745,5 @@ private fun AccelerationPlot() {
|
|||||||
@Preview
|
@Preview
|
||||||
@Composable
|
@Composable
|
||||||
fun HeadTrackingScreenPreview() {
|
fun HeadTrackingScreenPreview() {
|
||||||
HeadTrackingScreen(navController = NavController(LocalContext.current))
|
HeadTrackingScreen()
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.screens
|
package me.kavishdevar.librepods.screens
|
||||||
|
|
||||||
@@ -96,13 +96,10 @@ fun HearingAidAdjustmentsScreen(@Suppress("unused") navController: NavController
|
|||||||
val toneSliderValue = remember { mutableFloatStateOf(0.5f) }
|
val toneSliderValue = remember { mutableFloatStateOf(0.5f) }
|
||||||
val ambientNoiseReductionSliderValue = remember { mutableFloatStateOf(0.0f) }
|
val ambientNoiseReductionSliderValue = remember { mutableFloatStateOf(0.0f) }
|
||||||
val conversationBoostEnabled = remember { mutableStateOf(false) }
|
val conversationBoostEnabled = remember { mutableStateOf(false) }
|
||||||
val eq = remember { mutableStateOf(FloatArray(8)) }
|
val leftEQ = remember { mutableStateOf(FloatArray(8)) }
|
||||||
|
val rightEQ = remember { mutableStateOf(FloatArray(8)) }
|
||||||
val ownVoiceAmplification = remember { mutableFloatStateOf(0.5f) }
|
val ownVoiceAmplification = remember { mutableFloatStateOf(0.5f) }
|
||||||
|
|
||||||
val phoneMediaEQ = remember { mutableStateOf(FloatArray(8) { 0.5f }) }
|
|
||||||
val phoneEQEnabled = remember { mutableStateOf(false) }
|
|
||||||
val mediaEQEnabled = remember { mutableStateOf(false) }
|
|
||||||
|
|
||||||
val initialLoadComplete = remember { mutableStateOf(false) }
|
val initialLoadComplete = remember { mutableStateOf(false) }
|
||||||
|
|
||||||
val initialReadSucceeded = remember { mutableStateOf(false) }
|
val initialReadSucceeded = remember { mutableStateOf(false) }
|
||||||
@@ -111,8 +108,8 @@ fun HearingAidAdjustmentsScreen(@Suppress("unused") navController: NavController
|
|||||||
val hearingAidSettings = remember {
|
val hearingAidSettings = remember {
|
||||||
mutableStateOf(
|
mutableStateOf(
|
||||||
HearingAidSettings(
|
HearingAidSettings(
|
||||||
leftEQ = eq.value,
|
leftEQ = leftEQ.value,
|
||||||
rightEQ = eq.value,
|
rightEQ = rightEQ.value,
|
||||||
leftAmplification = amplificationSliderValue.floatValue + (0.5f - balanceSliderValue.floatValue) * amplificationSliderValue.floatValue * 2,
|
leftAmplification = amplificationSliderValue.floatValue + (0.5f - balanceSliderValue.floatValue) * amplificationSliderValue.floatValue * 2,
|
||||||
rightAmplification = amplificationSliderValue.floatValue + (balanceSliderValue.floatValue - 0.5f) * amplificationSliderValue.floatValue * 2,
|
rightAmplification = amplificationSliderValue.floatValue + (balanceSliderValue.floatValue - 0.5f) * amplificationSliderValue.floatValue * 2,
|
||||||
leftTone = toneSliderValue.floatValue,
|
leftTone = toneSliderValue.floatValue,
|
||||||
@@ -157,7 +154,8 @@ fun HearingAidAdjustmentsScreen(@Suppress("unused") navController: NavController
|
|||||||
toneSliderValue.floatValue = parsed.leftTone
|
toneSliderValue.floatValue = parsed.leftTone
|
||||||
ambientNoiseReductionSliderValue.floatValue = parsed.leftAmbientNoiseReduction
|
ambientNoiseReductionSliderValue.floatValue = parsed.leftAmbientNoiseReduction
|
||||||
conversationBoostEnabled.value = parsed.leftConversationBoost
|
conversationBoostEnabled.value = parsed.leftConversationBoost
|
||||||
eq.value = parsed.leftEQ.copyOf()
|
leftEQ.value = parsed.leftEQ.copyOf()
|
||||||
|
rightEQ.value = parsed.rightEQ.copyOf()
|
||||||
ownVoiceAmplification.floatValue = parsed.ownVoiceAmplification
|
ownVoiceAmplification.floatValue = parsed.ownVoiceAmplification
|
||||||
Log.d(TAG, "Updated hearing aid settings from notification")
|
Log.d(TAG, "Updated hearing aid settings from notification")
|
||||||
} else {
|
} else {
|
||||||
@@ -192,8 +190,8 @@ fun HearingAidAdjustmentsScreen(@Suppress("unused") navController: NavController
|
|||||||
}
|
}
|
||||||
|
|
||||||
hearingAidSettings.value = HearingAidSettings(
|
hearingAidSettings.value = HearingAidSettings(
|
||||||
leftEQ = eq.value,
|
leftEQ = leftEQ.value,
|
||||||
rightEQ = eq.value,
|
rightEQ = rightEQ.value,
|
||||||
leftAmplification = amplificationSliderValue.floatValue + if (balanceSliderValue.floatValue < 0) -balanceSliderValue.floatValue else 0f,
|
leftAmplification = amplificationSliderValue.floatValue + if (balanceSliderValue.floatValue < 0) -balanceSliderValue.floatValue else 0f,
|
||||||
rightAmplification = amplificationSliderValue.floatValue + if (balanceSliderValue.floatValue > 0) balanceSliderValue.floatValue else 0f,
|
rightAmplification = amplificationSliderValue.floatValue + if (balanceSliderValue.floatValue > 0) balanceSliderValue.floatValue else 0f,
|
||||||
leftTone = toneSliderValue.floatValue,
|
leftTone = toneSliderValue.floatValue,
|
||||||
@@ -216,26 +214,6 @@ fun HearingAidAdjustmentsScreen(@Suppress("unused") navController: NavController
|
|||||||
attManager.enableNotifications(ATTHandles.HEARING_AID)
|
attManager.enableNotifications(ATTHandles.HEARING_AID)
|
||||||
attManager.registerListener(ATTHandles.HEARING_AID, hearingAidATTListener)
|
attManager.registerListener(ATTHandles.HEARING_AID, hearingAidATTListener)
|
||||||
|
|
||||||
try {
|
|
||||||
if (aacpManager != null) {
|
|
||||||
Log.d(TAG, "Found AACPManager, reading cached EQ data")
|
|
||||||
val aacpEQ = aacpManager.eqData
|
|
||||||
if (aacpEQ.isNotEmpty()) {
|
|
||||||
eq.value = aacpEQ.copyOf()
|
|
||||||
phoneMediaEQ.value = aacpEQ.copyOf()
|
|
||||||
phoneEQEnabled.value = aacpManager.eqOnPhone
|
|
||||||
mediaEQEnabled.value = aacpManager.eqOnMedia
|
|
||||||
Log.d(TAG, "Populated EQ from AACPManager: ${aacpEQ.toList()}")
|
|
||||||
} else {
|
|
||||||
Log.d(TAG, "AACPManager EQ data empty")
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
Log.d(TAG, "No AACPManager available")
|
|
||||||
}
|
|
||||||
} catch (e: Exception) {
|
|
||||||
Log.w(TAG, "Error reading EQ from AACPManager: ${e.message}")
|
|
||||||
}
|
|
||||||
|
|
||||||
var parsedSettings: HearingAidSettings? = null
|
var parsedSettings: HearingAidSettings? = null
|
||||||
for (attempt in 1..3) {
|
for (attempt in 1..3) {
|
||||||
initialReadAttempts.intValue = attempt
|
initialReadAttempts.intValue = attempt
|
||||||
@@ -261,7 +239,8 @@ fun HearingAidAdjustmentsScreen(@Suppress("unused") navController: NavController
|
|||||||
toneSliderValue.floatValue = parsedSettings.leftTone
|
toneSliderValue.floatValue = parsedSettings.leftTone
|
||||||
ambientNoiseReductionSliderValue.floatValue = parsedSettings.leftAmbientNoiseReduction
|
ambientNoiseReductionSliderValue.floatValue = parsedSettings.leftAmbientNoiseReduction
|
||||||
conversationBoostEnabled.value = parsedSettings.leftConversationBoost
|
conversationBoostEnabled.value = parsedSettings.leftConversationBoost
|
||||||
eq.value = parsedSettings.leftEQ.copyOf()
|
leftEQ.value = parsedSettings.leftEQ.copyOf()
|
||||||
|
rightEQ.value = parsedSettings.rightEQ.copyOf()
|
||||||
ownVoiceAmplification.floatValue = parsedSettings.ownVoiceAmplification
|
ownVoiceAmplification.floatValue = parsedSettings.ownVoiceAmplification
|
||||||
initialReadSucceeded.value = true
|
initialReadSucceeded.value = true
|
||||||
} else {
|
} else {
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.screens
|
package me.kavishdevar.librepods.screens
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.screens
|
package me.kavishdevar.librepods.screens
|
||||||
|
|
||||||
|
|||||||
@@ -1,643 +0,0 @@
|
|||||||
/*
|
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
|
||||||
*
|
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
|
||||||
* This program is free software: you can redistribute it and/or modify
|
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
|
||||||
*
|
|
||||||
* 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 Affero General Public License for more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
|
|
||||||
package me.kavishdevar.librepods.screens
|
|
||||||
|
|
||||||
import android.content.Context
|
|
||||||
import android.util.Log
|
|
||||||
import androidx.compose.animation.AnimatedContent
|
|
||||||
import androidx.compose.animation.core.animateFloatAsState
|
|
||||||
import androidx.compose.animation.fadeIn
|
|
||||||
import androidx.compose.animation.fadeOut
|
|
||||||
import androidx.compose.animation.togetherWith
|
|
||||||
import androidx.compose.foundation.isSystemInDarkTheme
|
|
||||||
import androidx.compose.foundation.layout.Arrangement
|
|
||||||
import androidx.compose.foundation.layout.Box
|
|
||||||
import androidx.compose.foundation.layout.Column
|
|
||||||
import androidx.compose.foundation.layout.Spacer
|
|
||||||
import androidx.compose.foundation.layout.fillMaxSize
|
|
||||||
import androidx.compose.foundation.layout.fillMaxWidth
|
|
||||||
import androidx.compose.foundation.layout.height
|
|
||||||
import androidx.compose.foundation.layout.padding
|
|
||||||
import androidx.compose.foundation.layout.size
|
|
||||||
import androidx.compose.foundation.shape.RoundedCornerShape
|
|
||||||
import androidx.compose.material.icons.Icons
|
|
||||||
import androidx.compose.material.icons.filled.Check
|
|
||||||
import androidx.compose.material.icons.filled.Clear
|
|
||||||
import androidx.compose.material.icons.filled.Settings
|
|
||||||
import androidx.compose.material3.AlertDialog
|
|
||||||
import androidx.compose.material3.Button
|
|
||||||
import androidx.compose.material3.ButtonDefaults
|
|
||||||
import androidx.compose.material3.Card
|
|
||||||
import androidx.compose.material3.CardDefaults
|
|
||||||
import androidx.compose.material3.CircularProgressIndicator
|
|
||||||
import androidx.compose.material3.ExperimentalMaterial3Api
|
|
||||||
import androidx.compose.material3.Icon
|
|
||||||
import androidx.compose.material3.LinearProgressIndicator
|
|
||||||
import androidx.compose.material3.Text
|
|
||||||
import androidx.compose.material3.TextButton
|
|
||||||
import androidx.compose.runtime.Composable
|
|
||||||
import androidx.compose.runtime.LaunchedEffect
|
|
||||||
import androidx.compose.runtime.collectAsState
|
|
||||||
import androidx.compose.runtime.getValue
|
|
||||||
import androidx.compose.runtime.mutableStateOf
|
|
||||||
import androidx.compose.runtime.remember
|
|
||||||
import androidx.compose.runtime.setValue
|
|
||||||
import androidx.compose.ui.Alignment
|
|
||||||
import androidx.compose.ui.Modifier
|
|
||||||
import androidx.compose.ui.graphics.Color
|
|
||||||
import androidx.compose.ui.graphics.StrokeCap
|
|
||||||
import androidx.compose.ui.platform.LocalContext
|
|
||||||
import androidx.compose.ui.res.stringResource
|
|
||||||
import androidx.compose.ui.text.TextStyle
|
|
||||||
import androidx.compose.ui.text.font.Font
|
|
||||||
import androidx.compose.ui.text.font.FontFamily
|
|
||||||
import androidx.compose.ui.text.font.FontWeight
|
|
||||||
import androidx.compose.ui.text.style.TextAlign
|
|
||||||
import androidx.compose.ui.tooling.preview.Preview
|
|
||||||
import androidx.compose.ui.unit.dp
|
|
||||||
import androidx.compose.ui.unit.sp
|
|
||||||
import androidx.core.content.edit
|
|
||||||
import androidx.navigation.NavController
|
|
||||||
import com.kyant.backdrop.backdrops.layerBackdrop
|
|
||||||
import com.kyant.backdrop.backdrops.rememberLayerBackdrop
|
|
||||||
import dev.chrisbanes.haze.materials.ExperimentalHazeMaterialsApi
|
|
||||||
import kotlinx.coroutines.Dispatchers
|
|
||||||
import kotlinx.coroutines.launch
|
|
||||||
import kotlinx.coroutines.withContext
|
|
||||||
import me.kavishdevar.librepods.R
|
|
||||||
import me.kavishdevar.librepods.composables.StyledIconButton
|
|
||||||
import me.kavishdevar.librepods.composables.StyledScaffold
|
|
||||||
import me.kavishdevar.librepods.utils.RadareOffsetFinder
|
|
||||||
|
|
||||||
@ExperimentalHazeMaterialsApi
|
|
||||||
@OptIn(ExperimentalMaterial3Api::class)
|
|
||||||
@Composable
|
|
||||||
fun Onboarding(navController: NavController, activityContext: Context) {
|
|
||||||
val isDarkTheme = isSystemInDarkTheme()
|
|
||||||
val backgroundColor = if (isDarkTheme) Color(0xFF1C1C1E) else Color.White
|
|
||||||
val textColor = if (isDarkTheme) Color.White else Color.Black
|
|
||||||
val accentColor = if (isDarkTheme) Color(0xFF007AFF) else Color(0xFF3C6DF5)
|
|
||||||
|
|
||||||
val radareOffsetFinder = remember { RadareOffsetFinder(activityContext) }
|
|
||||||
val progressState by radareOffsetFinder.progressState.collectAsState()
|
|
||||||
var isComplete by remember { mutableStateOf(false) }
|
|
||||||
var hasStarted by remember { mutableStateOf(false) }
|
|
||||||
var rootCheckPassed by remember { mutableStateOf(false) }
|
|
||||||
var checkingRoot by remember { mutableStateOf(false) }
|
|
||||||
var rootCheckFailed by remember { mutableStateOf(false) }
|
|
||||||
var moduleEnabled by remember { mutableStateOf(false) }
|
|
||||||
var bluetoothToggled by remember { mutableStateOf(false) }
|
|
||||||
|
|
||||||
var showSkipDialog by remember { mutableStateOf(false) }
|
|
||||||
|
|
||||||
fun checkRootAccess() {
|
|
||||||
checkingRoot = true
|
|
||||||
rootCheckFailed = false
|
|
||||||
kotlinx.coroutines.MainScope().launch {
|
|
||||||
withContext(Dispatchers.IO) {
|
|
||||||
try {
|
|
||||||
val process = Runtime.getRuntime().exec("su -c id")
|
|
||||||
val exitValue = process.waitFor() // no idea why i have this, probably don't need to do this
|
|
||||||
withContext(Dispatchers.Main) {
|
|
||||||
rootCheckPassed = (exitValue == 0)
|
|
||||||
rootCheckFailed = (exitValue != 0)
|
|
||||||
checkingRoot = false
|
|
||||||
}
|
|
||||||
} catch (e: Exception) {
|
|
||||||
Log.e("Onboarding", "Root check failed", e)
|
|
||||||
withContext(Dispatchers.Main) {
|
|
||||||
rootCheckPassed = false
|
|
||||||
rootCheckFailed = true
|
|
||||||
checkingRoot = false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
LaunchedEffect(hasStarted) {
|
|
||||||
if (hasStarted && rootCheckPassed) {
|
|
||||||
Log.d("Onboarding", "Checking if hook offset is available...")
|
|
||||||
val isHookReady = radareOffsetFinder.isHookOffsetAvailable()
|
|
||||||
Log.d("Onboarding", "Hook offset ready: $isHookReady")
|
|
||||||
|
|
||||||
if (isHookReady) {
|
|
||||||
Log.d("Onboarding", "Hook is ready")
|
|
||||||
isComplete = true
|
|
||||||
} else {
|
|
||||||
Log.d("Onboarding", "Hook not ready, starting setup process...")
|
|
||||||
withContext(Dispatchers.IO) {
|
|
||||||
radareOffsetFinder.setupAndFindOffset()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
LaunchedEffect(progressState) {
|
|
||||||
if (progressState is RadareOffsetFinder.ProgressState.Success) {
|
|
||||||
isComplete = true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
val backdrop = rememberLayerBackdrop()
|
|
||||||
StyledScaffold(
|
|
||||||
title = "Setting Up",
|
|
||||||
actionButtons = listOf(
|
|
||||||
{scaffoldBackdrop ->
|
|
||||||
StyledIconButton(
|
|
||||||
onClick = {
|
|
||||||
showSkipDialog = true
|
|
||||||
},
|
|
||||||
icon = "",
|
|
||||||
darkMode = isDarkTheme,
|
|
||||||
backdrop = scaffoldBackdrop
|
|
||||||
)
|
|
||||||
}
|
|
||||||
)
|
|
||||||
) { spacerHeight ->
|
|
||||||
Column(
|
|
||||||
modifier = Modifier
|
|
||||||
.fillMaxSize()
|
|
||||||
.layerBackdrop(backdrop)
|
|
||||||
.padding(horizontal = 16.dp),
|
|
||||||
horizontalAlignment = Alignment.CenterHorizontally,
|
|
||||||
verticalArrangement = Arrangement.spacedBy(16.dp)
|
|
||||||
) {
|
|
||||||
Spacer(modifier = Modifier.height(spacerHeight))
|
|
||||||
|
|
||||||
Card(
|
|
||||||
modifier = Modifier.fillMaxWidth(),
|
|
||||||
colors = CardDefaults.cardColors(containerColor = backgroundColor),
|
|
||||||
shape = RoundedCornerShape(12.dp)
|
|
||||||
) {
|
|
||||||
Column(
|
|
||||||
modifier = Modifier
|
|
||||||
.fillMaxWidth()
|
|
||||||
.padding(24.dp),
|
|
||||||
horizontalAlignment = Alignment.CenterHorizontally
|
|
||||||
) {
|
|
||||||
if (!rootCheckPassed && !hasStarted) {
|
|
||||||
Icon(
|
|
||||||
imageVector = Icons.Default.Settings,
|
|
||||||
contentDescription = "Root Access",
|
|
||||||
tint = accentColor,
|
|
||||||
modifier = Modifier.size(50.dp)
|
|
||||||
)
|
|
||||||
|
|
||||||
Spacer(modifier = Modifier.height(24.dp))
|
|
||||||
|
|
||||||
Text(
|
|
||||||
text = stringResource(R.string.root_access_required),
|
|
||||||
style = TextStyle(
|
|
||||||
fontSize = 22.sp,
|
|
||||||
fontWeight = FontWeight.Bold,
|
|
||||||
textAlign = TextAlign.Center,
|
|
||||||
fontFamily = FontFamily(Font(R.font.sf_pro)),
|
|
||||||
color = textColor
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
Spacer(modifier = Modifier.height(8.dp))
|
|
||||||
|
|
||||||
Text(
|
|
||||||
text = stringResource(R.string.this_app_needs_root_access_to_hook_onto_the_bluetooth_library),
|
|
||||||
style = TextStyle(
|
|
||||||
fontSize = 16.sp,
|
|
||||||
fontWeight = FontWeight.Normal,
|
|
||||||
textAlign = TextAlign.Center,
|
|
||||||
fontFamily = FontFamily(Font(R.font.sf_pro)),
|
|
||||||
color = textColor.copy(alpha = 0.7f)
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
if (rootCheckFailed) {
|
|
||||||
Spacer(modifier = Modifier.height(8.dp))
|
|
||||||
Text(
|
|
||||||
text = stringResource(R.string.root_access_denied),
|
|
||||||
style = TextStyle(
|
|
||||||
fontSize = 14.sp,
|
|
||||||
fontWeight = FontWeight.Normal,
|
|
||||||
textAlign = TextAlign.Center,
|
|
||||||
fontFamily = FontFamily(Font(R.font.sf_pro)),
|
|
||||||
color = Color(0xFFFF453A)
|
|
||||||
)
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
Spacer(modifier = Modifier.height(24.dp))
|
|
||||||
|
|
||||||
Button(
|
|
||||||
onClick = { checkRootAccess() },
|
|
||||||
modifier = Modifier
|
|
||||||
.fillMaxWidth()
|
|
||||||
.height(50.dp),
|
|
||||||
colors = ButtonDefaults.buttonColors(
|
|
||||||
containerColor = accentColor
|
|
||||||
),
|
|
||||||
shape = RoundedCornerShape(8.dp),
|
|
||||||
enabled = !checkingRoot
|
|
||||||
) {
|
|
||||||
if (checkingRoot) {
|
|
||||||
CircularProgressIndicator(
|
|
||||||
modifier = Modifier.size(24.dp),
|
|
||||||
color = Color.White,
|
|
||||||
strokeWidth = 2.dp
|
|
||||||
)
|
|
||||||
} else {
|
|
||||||
Text(
|
|
||||||
"Check Root Access",
|
|
||||||
style = TextStyle(
|
|
||||||
fontSize = 16.sp,
|
|
||||||
fontWeight = FontWeight.Medium,
|
|
||||||
fontFamily = FontFamily(Font(R.font.sf_pro))
|
|
||||||
),
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
StatusIcon(if (hasStarted) progressState else RadareOffsetFinder.ProgressState.Idle, isDarkTheme)
|
|
||||||
|
|
||||||
Spacer(modifier = Modifier.height(24.dp))
|
|
||||||
|
|
||||||
AnimatedContent(
|
|
||||||
targetState = if (hasStarted) getStatusTitle(progressState,
|
|
||||||
moduleEnabled, bluetoothToggled) else "Setup Required",
|
|
||||||
transitionSpec = { fadeIn() togetherWith fadeOut() }
|
|
||||||
) { text ->
|
|
||||||
Text(
|
|
||||||
text = text,
|
|
||||||
style = TextStyle(
|
|
||||||
fontSize = 22.sp,
|
|
||||||
fontWeight = FontWeight.Bold,
|
|
||||||
textAlign = TextAlign.Center,
|
|
||||||
fontFamily = FontFamily(Font(R.font.sf_pro)),
|
|
||||||
color = textColor
|
|
||||||
)
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
Spacer(modifier = Modifier.height(8.dp))
|
|
||||||
|
|
||||||
AnimatedContent(
|
|
||||||
targetState = if (hasStarted)
|
|
||||||
getStatusDescription(progressState, moduleEnabled, bluetoothToggled)
|
|
||||||
else
|
|
||||||
"AirPods functionality requires one-time setup for hooking into Bluetooth library",
|
|
||||||
transitionSpec = { fadeIn() togetherWith fadeOut() }
|
|
||||||
) { text ->
|
|
||||||
Text(
|
|
||||||
text = text,
|
|
||||||
style = TextStyle(
|
|
||||||
fontSize = 16.sp,
|
|
||||||
fontWeight = FontWeight.Normal,
|
|
||||||
textAlign = TextAlign.Center,
|
|
||||||
fontFamily = FontFamily(Font(R.font.sf_pro)),
|
|
||||||
color = textColor.copy(alpha = 0.7f)
|
|
||||||
)
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
Spacer(modifier = Modifier.height(24.dp))
|
|
||||||
|
|
||||||
if (!hasStarted) {
|
|
||||||
Button(
|
|
||||||
onClick = { hasStarted = true },
|
|
||||||
modifier = Modifier
|
|
||||||
.fillMaxWidth()
|
|
||||||
.height(50.dp),
|
|
||||||
colors = ButtonDefaults.buttonColors(
|
|
||||||
containerColor = accentColor
|
|
||||||
),
|
|
||||||
shape = RoundedCornerShape(8.dp)
|
|
||||||
) {
|
|
||||||
Text(
|
|
||||||
"Start Setup",
|
|
||||||
style = TextStyle(
|
|
||||||
fontSize = 16.sp,
|
|
||||||
fontWeight = FontWeight.Medium,
|
|
||||||
fontFamily = FontFamily(Font(R.font.sf_pro))
|
|
||||||
),
|
|
||||||
)
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
when (progressState) {
|
|
||||||
is RadareOffsetFinder.ProgressState.DownloadProgress -> {
|
|
||||||
val progress = (progressState as RadareOffsetFinder.ProgressState.DownloadProgress).progress
|
|
||||||
val animatedProgress by animateFloatAsState(
|
|
||||||
targetValue = progress,
|
|
||||||
label = "Download Progress"
|
|
||||||
)
|
|
||||||
Column(
|
|
||||||
modifier = Modifier.fillMaxWidth(),
|
|
||||||
horizontalAlignment = Alignment.CenterHorizontally
|
|
||||||
) {
|
|
||||||
LinearProgressIndicator(
|
|
||||||
progress = { animatedProgress },
|
|
||||||
modifier = Modifier
|
|
||||||
.fillMaxWidth()
|
|
||||||
.height(8.dp),
|
|
||||||
strokeCap = StrokeCap.Round,
|
|
||||||
color = accentColor
|
|
||||||
)
|
|
||||||
|
|
||||||
Spacer(modifier = Modifier.height(8.dp))
|
|
||||||
|
|
||||||
Text(
|
|
||||||
text = "${(progress * 100).toInt()}%",
|
|
||||||
style = TextStyle(
|
|
||||||
fontSize = 14.sp,
|
|
||||||
fontWeight = FontWeight.Medium,
|
|
||||||
fontFamily = FontFamily(Font(R.font.sf_pro)),
|
|
||||||
color = textColor.copy(alpha = 0.6f)
|
|
||||||
)
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
is RadareOffsetFinder.ProgressState.Success -> {
|
|
||||||
if (!moduleEnabled) {
|
|
||||||
Button(
|
|
||||||
onClick = { moduleEnabled = true },
|
|
||||||
modifier = Modifier
|
|
||||||
.fillMaxWidth()
|
|
||||||
.height(50.dp),
|
|
||||||
colors = ButtonDefaults.buttonColors(
|
|
||||||
containerColor = accentColor
|
|
||||||
),
|
|
||||||
shape = RoundedCornerShape(8.dp)
|
|
||||||
) {
|
|
||||||
Text(
|
|
||||||
"I've Enabled/Reactivated the Module",
|
|
||||||
style = TextStyle(
|
|
||||||
fontSize = 16.sp,
|
|
||||||
fontWeight = FontWeight.Medium,
|
|
||||||
fontFamily = FontFamily(Font(R.font.sf_pro))
|
|
||||||
),
|
|
||||||
)
|
|
||||||
}
|
|
||||||
} else if (!bluetoothToggled) {
|
|
||||||
Button(
|
|
||||||
onClick = { bluetoothToggled = true },
|
|
||||||
modifier = Modifier
|
|
||||||
.fillMaxWidth()
|
|
||||||
.height(50.dp),
|
|
||||||
colors = ButtonDefaults.buttonColors(
|
|
||||||
containerColor = accentColor
|
|
||||||
),
|
|
||||||
shape = RoundedCornerShape(8.dp)
|
|
||||||
) {
|
|
||||||
Text(
|
|
||||||
"I've Toggled Bluetooth",
|
|
||||||
style = TextStyle(
|
|
||||||
fontSize = 16.sp,
|
|
||||||
fontWeight = FontWeight.Medium,
|
|
||||||
fontFamily = FontFamily(Font(R.font.sf_pro))
|
|
||||||
),
|
|
||||||
)
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
Button(
|
|
||||||
onClick = {
|
|
||||||
navController.navigate("settings") {
|
|
||||||
popUpTo("onboarding") { inclusive = true }
|
|
||||||
}
|
|
||||||
},
|
|
||||||
modifier = Modifier
|
|
||||||
.fillMaxWidth()
|
|
||||||
.height(50.dp),
|
|
||||||
colors = ButtonDefaults.buttonColors(
|
|
||||||
containerColor = accentColor
|
|
||||||
),
|
|
||||||
shape = RoundedCornerShape(8.dp)
|
|
||||||
) {
|
|
||||||
Text(
|
|
||||||
"Continue to Settings",
|
|
||||||
style = TextStyle(
|
|
||||||
fontSize = 16.sp,
|
|
||||||
fontWeight = FontWeight.Medium,
|
|
||||||
fontFamily = FontFamily(Font(R.font.sf_pro))
|
|
||||||
),
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
is RadareOffsetFinder.ProgressState.Idle,
|
|
||||||
is RadareOffsetFinder.ProgressState.Error -> {
|
|
||||||
// No specific UI for these states
|
|
||||||
}
|
|
||||||
else -> {
|
|
||||||
LinearProgressIndicator(
|
|
||||||
modifier = Modifier
|
|
||||||
.fillMaxWidth()
|
|
||||||
.height(8.dp),
|
|
||||||
strokeCap = StrokeCap.Round,
|
|
||||||
color = accentColor
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Spacer(modifier = Modifier.weight(1f))
|
|
||||||
|
|
||||||
if (progressState is RadareOffsetFinder.ProgressState.Error && !isComplete && hasStarted) {
|
|
||||||
Button(
|
|
||||||
onClick = {
|
|
||||||
Log.d("Onboarding", "Trying to find offset again...")
|
|
||||||
kotlinx.coroutines.MainScope().launch {
|
|
||||||
withContext(Dispatchers.IO) {
|
|
||||||
radareOffsetFinder.setupAndFindOffset()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
modifier = Modifier
|
|
||||||
.fillMaxWidth()
|
|
||||||
.height(55.dp),
|
|
||||||
colors = ButtonDefaults.buttonColors(
|
|
||||||
containerColor = accentColor
|
|
||||||
),
|
|
||||||
shape = RoundedCornerShape(8.dp)
|
|
||||||
) {
|
|
||||||
Text(
|
|
||||||
"Try Again",
|
|
||||||
style = TextStyle(
|
|
||||||
fontSize = 16.sp,
|
|
||||||
fontWeight = FontWeight.Medium,
|
|
||||||
fontFamily = FontFamily(Font(R.font.sf_pro))
|
|
||||||
),
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (showSkipDialog) {
|
|
||||||
AlertDialog(
|
|
||||||
onDismissRequest = { showSkipDialog = false },
|
|
||||||
title = { Text("Skip Setup") },
|
|
||||||
text = {
|
|
||||||
Text(
|
|
||||||
"Have you installed the root module that patches the Bluetooth library directly? This option is for users who have manually patched their system instead of using the dynamic hook.",
|
|
||||||
style = TextStyle(
|
|
||||||
fontSize = 16.sp,
|
|
||||||
fontFamily = FontFamily(Font(R.font.sf_pro))
|
|
||||||
)
|
|
||||||
)
|
|
||||||
},
|
|
||||||
confirmButton = {
|
|
||||||
val sharedPreferences = activityContext.getSharedPreferences("settings", Context.MODE_PRIVATE)
|
|
||||||
TextButton(
|
|
||||||
onClick = {
|
|
||||||
showSkipDialog = false
|
|
||||||
RadareOffsetFinder.clearHookOffsets()
|
|
||||||
sharedPreferences.edit { putBoolean("skip_setup", true) }
|
|
||||||
navController.navigate("settings") {
|
|
||||||
popUpTo("onboarding") { inclusive = true }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
) {
|
|
||||||
Text(
|
|
||||||
"Yes, Skip Setup",
|
|
||||||
color = accentColor,
|
|
||||||
fontWeight = FontWeight.Bold
|
|
||||||
)
|
|
||||||
}
|
|
||||||
},
|
|
||||||
dismissButton = {
|
|
||||||
TextButton(
|
|
||||||
onClick = { showSkipDialog = false }
|
|
||||||
) {
|
|
||||||
Text("Cancel")
|
|
||||||
}
|
|
||||||
},
|
|
||||||
containerColor = backgroundColor,
|
|
||||||
textContentColor = textColor,
|
|
||||||
titleContentColor = textColor
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Composable
|
|
||||||
private fun StatusIcon(
|
|
||||||
progressState: RadareOffsetFinder.ProgressState,
|
|
||||||
isDarkTheme: Boolean
|
|
||||||
) {
|
|
||||||
val accentColor = if (isDarkTheme) Color(0xFF007AFF) else Color(0xFF3C6DF5)
|
|
||||||
val errorColor = if (isDarkTheme) Color(0xFFFF453A) else Color(0xFFFF3B30)
|
|
||||||
val successColor = if (isDarkTheme) Color(0xFF30D158) else Color(0xFF34C759)
|
|
||||||
|
|
||||||
Box(
|
|
||||||
modifier = Modifier.size(80.dp),
|
|
||||||
contentAlignment = Alignment.Center
|
|
||||||
) {
|
|
||||||
when (progressState) {
|
|
||||||
is RadareOffsetFinder.ProgressState.Error -> {
|
|
||||||
Icon(
|
|
||||||
imageVector = Icons.Default.Clear,
|
|
||||||
contentDescription = "Error",
|
|
||||||
tint = errorColor,
|
|
||||||
modifier = Modifier.size(50.dp)
|
|
||||||
)
|
|
||||||
}
|
|
||||||
is RadareOffsetFinder.ProgressState.Success -> {
|
|
||||||
Icon(
|
|
||||||
imageVector = Icons.Default.Check,
|
|
||||||
contentDescription = "Success",
|
|
||||||
tint = successColor,
|
|
||||||
modifier = Modifier.size(50.dp)
|
|
||||||
)
|
|
||||||
}
|
|
||||||
is RadareOffsetFinder.ProgressState.Idle -> {
|
|
||||||
Icon(
|
|
||||||
imageVector = Icons.Default.Settings,
|
|
||||||
contentDescription = "Settings",
|
|
||||||
tint = accentColor,
|
|
||||||
modifier = Modifier.size(50.dp)
|
|
||||||
)
|
|
||||||
}
|
|
||||||
else -> {
|
|
||||||
CircularProgressIndicator(
|
|
||||||
modifier = Modifier.size(50.dp),
|
|
||||||
color = accentColor,
|
|
||||||
strokeWidth = 4.dp
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun getStatusTitle(
|
|
||||||
state: RadareOffsetFinder.ProgressState,
|
|
||||||
moduleEnabled: Boolean,
|
|
||||||
bluetoothToggled: Boolean
|
|
||||||
): String {
|
|
||||||
return when (state) {
|
|
||||||
is RadareOffsetFinder.ProgressState.Success -> {
|
|
||||||
when {
|
|
||||||
!moduleEnabled -> "Enable Xposed Module"
|
|
||||||
!bluetoothToggled -> "Toggle Bluetooth"
|
|
||||||
else -> "Setup Complete"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
is RadareOffsetFinder.ProgressState.Idle -> "Getting Ready"
|
|
||||||
is RadareOffsetFinder.ProgressState.CheckingExisting -> "Checking if radare2 already downloaded"
|
|
||||||
is RadareOffsetFinder.ProgressState.Downloading -> "Downloading radare2"
|
|
||||||
is RadareOffsetFinder.ProgressState.DownloadProgress -> "Downloading radare2"
|
|
||||||
is RadareOffsetFinder.ProgressState.Extracting -> "Extracting radare2"
|
|
||||||
is RadareOffsetFinder.ProgressState.MakingExecutable -> "Setting executable permissions"
|
|
||||||
is RadareOffsetFinder.ProgressState.FindingOffset -> "Finding function offset"
|
|
||||||
is RadareOffsetFinder.ProgressState.SavingOffset -> "Saving offset"
|
|
||||||
is RadareOffsetFinder.ProgressState.Cleaning -> "Cleaning Up"
|
|
||||||
is RadareOffsetFinder.ProgressState.Error -> "Setup Failed"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun getStatusDescription(
|
|
||||||
state: RadareOffsetFinder.ProgressState,
|
|
||||||
moduleEnabled: Boolean,
|
|
||||||
bluetoothToggled: Boolean
|
|
||||||
): String {
|
|
||||||
return when (state) {
|
|
||||||
is RadareOffsetFinder.ProgressState.Success -> {
|
|
||||||
when {
|
|
||||||
!moduleEnabled -> "Please enable the LibrePods Xposed module in your Xposed manager (e.g. LSPosed). If already enabled, disable and re-enable it."
|
|
||||||
!bluetoothToggled -> "Please turn off and then turn on Bluetooth to apply the changes."
|
|
||||||
else -> "All set! You can now use your AirPods with enhanced functionality."
|
|
||||||
}
|
|
||||||
}
|
|
||||||
is RadareOffsetFinder.ProgressState.Idle -> "Preparing"
|
|
||||||
is RadareOffsetFinder.ProgressState.CheckingExisting -> "Checking if radare2 are already installed"
|
|
||||||
is RadareOffsetFinder.ProgressState.Downloading -> "Starting radare2 download"
|
|
||||||
is RadareOffsetFinder.ProgressState.DownloadProgress -> "Downloading radare2"
|
|
||||||
is RadareOffsetFinder.ProgressState.Extracting -> "Extracting radare2"
|
|
||||||
is RadareOffsetFinder.ProgressState.MakingExecutable -> "Setting executable permissions on radare2 binaries"
|
|
||||||
is RadareOffsetFinder.ProgressState.FindingOffset -> "Looking for the required Bluetooth function in system libraries"
|
|
||||||
is RadareOffsetFinder.ProgressState.SavingOffset -> "Saving the function offset"
|
|
||||||
is RadareOffsetFinder.ProgressState.Cleaning -> "Removing temporary extracted files"
|
|
||||||
is RadareOffsetFinder.ProgressState.Error -> state.message
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@ExperimentalHazeMaterialsApi
|
|
||||||
@Preview
|
|
||||||
@Composable
|
|
||||||
fun OnboardingPreview() {
|
|
||||||
Onboarding(navController = NavController(LocalContext.current), activityContext = LocalContext.current)
|
|
||||||
}
|
|
||||||
|
|
||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.screens
|
package me.kavishdevar.librepods.screens
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@file:OptIn(ExperimentalStdlibApi::class, ExperimentalEncodingApi::class)
|
@file:OptIn(ExperimentalStdlibApi::class, ExperimentalEncodingApi::class)
|
||||||
|
|
||||||
@@ -186,7 +186,7 @@ fun LongPress(navController: NavController, name: String) {
|
|||||||
listeningModeItems.add(
|
listeningModeItems.add(
|
||||||
SelectItem(
|
SelectItem(
|
||||||
name = stringResource(R.string.off),
|
name = stringResource(R.string.off),
|
||||||
description = "Turns off noise management",
|
description = stringResource(R.string.listening_mode_off_description),
|
||||||
iconRes = R.drawable.noise_cancellation,
|
iconRes = R.drawable.noise_cancellation,
|
||||||
selected = (currentByte and 0x01) != 0,
|
selected = (currentByte and 0x01) != 0,
|
||||||
onClick = {
|
onClick = {
|
||||||
@@ -212,7 +212,7 @@ fun LongPress(navController: NavController, name: String) {
|
|||||||
listeningModeItems.addAll(listOf(
|
listeningModeItems.addAll(listOf(
|
||||||
SelectItem(
|
SelectItem(
|
||||||
name = stringResource(R.string.transparency),
|
name = stringResource(R.string.transparency),
|
||||||
description = "Lets in external sounds",
|
description = stringResource(R.string.listening_mode_transparency_description),
|
||||||
iconRes = R.drawable.transparency,
|
iconRes = R.drawable.transparency,
|
||||||
selected = (currentByte and 0x04) != 0,
|
selected = (currentByte and 0x04) != 0,
|
||||||
onClick = {
|
onClick = {
|
||||||
@@ -235,7 +235,7 @@ fun LongPress(navController: NavController, name: String) {
|
|||||||
),
|
),
|
||||||
SelectItem(
|
SelectItem(
|
||||||
name = stringResource(R.string.adaptive),
|
name = stringResource(R.string.adaptive),
|
||||||
description = "Dynamically adjust external noise",
|
description = stringResource(R.string.listening_mode_adaptive_description),
|
||||||
iconRes = R.drawable.adaptive,
|
iconRes = R.drawable.adaptive,
|
||||||
selected = (currentByte and 0x08) != 0,
|
selected = (currentByte and 0x08) != 0,
|
||||||
onClick = {
|
onClick = {
|
||||||
@@ -258,7 +258,7 @@ fun LongPress(navController: NavController, name: String) {
|
|||||||
),
|
),
|
||||||
SelectItem(
|
SelectItem(
|
||||||
name = stringResource(R.string.noise_cancellation),
|
name = stringResource(R.string.noise_cancellation),
|
||||||
description = "Blocks out external sounds",
|
description = stringResource(R.string.listening_mode_noise_cancellation_description),
|
||||||
iconRes = R.drawable.noise_cancellation,
|
iconRes = R.drawable.noise_cancellation,
|
||||||
selected = (currentByte and 0x02) != 0,
|
selected = (currentByte and 0x02) != 0,
|
||||||
onClick = {
|
onClick = {
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@file:OptIn(ExperimentalEncodingApi::class)
|
@file:OptIn(ExperimentalEncodingApi::class)
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.screens
|
package me.kavishdevar.librepods.screens
|
||||||
|
|
||||||
@@ -71,7 +71,7 @@ import me.kavishdevar.librepods.composables.StyledSlider
|
|||||||
import me.kavishdevar.librepods.composables.StyledToggle
|
import me.kavishdevar.librepods.composables.StyledToggle
|
||||||
import me.kavishdevar.librepods.services.ServiceManager
|
import me.kavishdevar.librepods.services.ServiceManager
|
||||||
import me.kavishdevar.librepods.utils.ATTHandles
|
import me.kavishdevar.librepods.utils.ATTHandles
|
||||||
import me.kavishdevar.librepods.utils.RadareOffsetFinder
|
// import me.kavishdevar.librepods.utils.RadareOffsetFinder
|
||||||
import me.kavishdevar.librepods.utils.TransparencySettings
|
import me.kavishdevar.librepods.utils.TransparencySettings
|
||||||
import me.kavishdevar.librepods.utils.parseTransparencySettingsResponse
|
import me.kavishdevar.librepods.utils.parseTransparencySettingsResponse
|
||||||
import me.kavishdevar.librepods.utils.sendTransparencySettings
|
import me.kavishdevar.librepods.utils.sendTransparencySettings
|
||||||
@@ -90,8 +90,8 @@ fun TransparencySettingsScreen(navController: NavController) {
|
|||||||
val verticalScrollState = rememberScrollState()
|
val verticalScrollState = rememberScrollState()
|
||||||
val attManager = ServiceManager.getService()?.attManager ?: return
|
val attManager = ServiceManager.getService()?.attManager ?: return
|
||||||
val aacpManager = remember { ServiceManager.getService()?.aacpManager }
|
val aacpManager = remember { ServiceManager.getService()?.aacpManager }
|
||||||
val isSdpOffsetAvailable =
|
val isSdpOffsetAvailable = remember { mutableStateOf(false) } // always available rn, for testing without radare
|
||||||
remember { mutableStateOf(RadareOffsetFinder.isSdpOffsetAvailable()) }
|
// remember { mutableStateOf(RadareOffsetFinder.isSdpOffsetAvailable()) }
|
||||||
|
|
||||||
val trackColor = if (isDarkTheme) Color(0xFFB3B3B3) else Color(0xFF929491)
|
val trackColor = if (isDarkTheme) Color(0xFFB3B3B3) else Color(0xFF929491)
|
||||||
val activeTrackColor = if (isDarkTheme) Color(0xFF007AFF) else Color(0xFF3C6DF5)
|
val activeTrackColor = if (isDarkTheme) Color(0xFF007AFF) else Color(0xFF3C6DF5)
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.screens
|
package me.kavishdevar.librepods.screens
|
||||||
|
|
||||||
|
|||||||
@@ -1,25 +1,26 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.screens
|
package me.kavishdevar.librepods.screens
|
||||||
|
|
||||||
import android.annotation.SuppressLint
|
import android.annotation.SuppressLint
|
||||||
import android.util.Log
|
import android.util.Log
|
||||||
|
import androidx.compose.foundation.isSystemInDarkTheme
|
||||||
import androidx.compose.foundation.layout.Arrangement
|
import androidx.compose.foundation.layout.Arrangement
|
||||||
import androidx.compose.foundation.layout.Column
|
import androidx.compose.foundation.layout.Column
|
||||||
import androidx.compose.foundation.layout.Row
|
import androidx.compose.foundation.layout.Row
|
||||||
@@ -39,11 +40,13 @@ import androidx.compose.runtime.Composable
|
|||||||
import androidx.compose.runtime.DisposableEffect
|
import androidx.compose.runtime.DisposableEffect
|
||||||
import androidx.compose.runtime.LaunchedEffect
|
import androidx.compose.runtime.LaunchedEffect
|
||||||
import androidx.compose.runtime.MutableState
|
import androidx.compose.runtime.MutableState
|
||||||
|
import androidx.compose.runtime.mutableFloatStateOf
|
||||||
import androidx.compose.runtime.mutableIntStateOf
|
import androidx.compose.runtime.mutableIntStateOf
|
||||||
import androidx.compose.runtime.mutableStateOf
|
import androidx.compose.runtime.mutableStateOf
|
||||||
import androidx.compose.runtime.remember
|
import androidx.compose.runtime.remember
|
||||||
import androidx.compose.ui.Alignment
|
import androidx.compose.ui.Alignment
|
||||||
import androidx.compose.ui.Modifier
|
import androidx.compose.ui.Modifier
|
||||||
|
import androidx.compose.ui.graphics.Color
|
||||||
import androidx.compose.ui.res.stringResource
|
import androidx.compose.ui.res.stringResource
|
||||||
import androidx.compose.ui.text.TextStyle
|
import androidx.compose.ui.text.TextStyle
|
||||||
import androidx.compose.ui.text.font.Font
|
import androidx.compose.ui.text.font.Font
|
||||||
@@ -62,7 +65,6 @@ import kotlinx.coroutines.delay
|
|||||||
import me.kavishdevar.librepods.R
|
import me.kavishdevar.librepods.R
|
||||||
import me.kavishdevar.librepods.composables.StyledScaffold
|
import me.kavishdevar.librepods.composables.StyledScaffold
|
||||||
import me.kavishdevar.librepods.services.ServiceManager
|
import me.kavishdevar.librepods.services.ServiceManager
|
||||||
import me.kavishdevar.librepods.utils.AACPManager
|
|
||||||
import me.kavishdevar.librepods.utils.ATTHandles
|
import me.kavishdevar.librepods.utils.ATTHandles
|
||||||
import me.kavishdevar.librepods.utils.HearingAidSettings
|
import me.kavishdevar.librepods.utils.HearingAidSettings
|
||||||
import me.kavishdevar.librepods.utils.parseHearingAidSettingsResponse
|
import me.kavishdevar.librepods.utils.parseHearingAidSettingsResponse
|
||||||
@@ -91,7 +93,6 @@ fun UpdateHearingTestScreen(@Suppress("unused") navController: NavController) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
val aacpManager = remember { ServiceManager.getService()?.aacpManager }
|
|
||||||
val backdrop = rememberLayerBackdrop()
|
val backdrop = rememberLayerBackdrop()
|
||||||
StyledScaffold(
|
StyledScaffold(
|
||||||
title = stringResource(R.string.hearing_test)
|
title = stringResource(R.string.hearing_test)
|
||||||
@@ -105,16 +106,25 @@ fun UpdateHearingTestScreen(@Suppress("unused") navController: NavController) {
|
|||||||
.padding(horizontal = 16.dp),
|
.padding(horizontal = 16.dp),
|
||||||
verticalArrangement = Arrangement.spacedBy(16.dp)
|
verticalArrangement = Arrangement.spacedBy(16.dp)
|
||||||
) {
|
) {
|
||||||
|
val textColor = if (isSystemInDarkTheme()) Color.White else Color.Black
|
||||||
|
|
||||||
Spacer(modifier = Modifier.height(spacerHeight))
|
Spacer(modifier = Modifier.height(spacerHeight))
|
||||||
|
|
||||||
Text(
|
Text(
|
||||||
text = stringResource(R.string.hearing_test_value_instruction),
|
text = stringResource(R.string.hearing_test_value_instruction),
|
||||||
fontSize = 16.sp,
|
|
||||||
modifier = Modifier.fillMaxWidth(),
|
modifier = Modifier.fillMaxWidth(),
|
||||||
|
style = TextStyle(
|
||||||
|
fontSize = 16.sp,
|
||||||
|
color = textColor,
|
||||||
|
fontFamily = FontFamily(Font(R.font.sf_pro))
|
||||||
|
),
|
||||||
textAlign = TextAlign.Center,
|
textAlign = TextAlign.Center,
|
||||||
fontFamily = FontFamily(Font(R.font.sf_pro))
|
|
||||||
)
|
)
|
||||||
|
val tone = remember { mutableFloatStateOf(0.5f) }
|
||||||
|
val ambientNoiseReduction = remember { mutableFloatStateOf(0.0f) }
|
||||||
|
val ownVoiceAmplification = remember { mutableFloatStateOf(0.5f) }
|
||||||
|
val leftAmplification = remember { mutableFloatStateOf(0.5f) }
|
||||||
|
val rightAmplification = remember { mutableFloatStateOf(0.5f) }
|
||||||
val conversationBoostEnabled = remember { mutableStateOf(false) }
|
val conversationBoostEnabled = remember { mutableStateOf(false) }
|
||||||
val leftEQ = remember { mutableStateOf(FloatArray(8)) }
|
val leftEQ = remember { mutableStateOf(FloatArray(8)) }
|
||||||
val rightEQ = remember { mutableStateOf(FloatArray(8)) }
|
val rightEQ = remember { mutableStateOf(FloatArray(8)) }
|
||||||
@@ -128,40 +138,21 @@ fun UpdateHearingTestScreen(@Suppress("unused") navController: NavController) {
|
|||||||
HearingAidSettings(
|
HearingAidSettings(
|
||||||
leftEQ = leftEQ.value,
|
leftEQ = leftEQ.value,
|
||||||
rightEQ = rightEQ.value,
|
rightEQ = rightEQ.value,
|
||||||
leftAmplification = 0.5f,
|
leftAmplification = leftAmplification.value,
|
||||||
rightAmplification = 0.5f,
|
rightAmplification = rightAmplification.value,
|
||||||
leftTone = 0.5f,
|
leftTone = tone.value,
|
||||||
rightTone = 0.5f,
|
rightTone = tone.value,
|
||||||
leftConversationBoost = conversationBoostEnabled.value,
|
leftConversationBoost = conversationBoostEnabled.value,
|
||||||
rightConversationBoost = conversationBoostEnabled.value,
|
rightConversationBoost = conversationBoostEnabled.value,
|
||||||
leftAmbientNoiseReduction = 0.0f,
|
leftAmbientNoiseReduction = ambientNoiseReduction.value,
|
||||||
rightAmbientNoiseReduction = 0.0f,
|
rightAmbientNoiseReduction = ambientNoiseReduction.value,
|
||||||
netAmplification = 0.5f,
|
netAmplification = leftAmplification.value + rightAmplification.value / 2,
|
||||||
balance = 0.5f,
|
balance = 0.5f + (rightAmplification.value - leftAmplification.value) / 2,
|
||||||
ownVoiceAmplification = 0.5f
|
ownVoiceAmplification = ownVoiceAmplification.value
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
val hearingAidEnabled = remember {
|
|
||||||
val aidStatus = aacpManager?.controlCommandStatusList?.find { it.identifier == AACPManager.Companion.ControlCommandIdentifiers.HEARING_AID }
|
|
||||||
val assistStatus = aacpManager?.controlCommandStatusList?.find { it.identifier == AACPManager.Companion.ControlCommandIdentifiers.HEARING_ASSIST_CONFIG }
|
|
||||||
mutableStateOf((aidStatus?.value?.getOrNull(1) == 0x01.toByte()) && (assistStatus?.value?.getOrNull(0) == 0x01.toByte()))
|
|
||||||
}
|
|
||||||
|
|
||||||
val hearingAidListener = remember {
|
|
||||||
object : AACPManager.ControlCommandListener {
|
|
||||||
override fun onControlCommandReceived(controlCommand: AACPManager.ControlCommand) {
|
|
||||||
if (controlCommand.identifier == AACPManager.Companion.ControlCommandIdentifiers.HEARING_AID.value ||
|
|
||||||
controlCommand.identifier == AACPManager.Companion.ControlCommandIdentifiers.HEARING_ASSIST_CONFIG.value) {
|
|
||||||
val aidStatus = aacpManager?.controlCommandStatusList?.find { it.identifier == AACPManager.Companion.ControlCommandIdentifiers.HEARING_AID }
|
|
||||||
val assistStatus = aacpManager?.controlCommandStatusList?.find { it.identifier == AACPManager.Companion.ControlCommandIdentifiers.HEARING_ASSIST_CONFIG }
|
|
||||||
hearingAidEnabled.value = (aidStatus?.value?.getOrNull(1) == 0x01.toByte()) && (assistStatus?.value?.getOrNull(0) == 0x01.toByte())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
val hearingAidATTListener = remember {
|
val hearingAidATTListener = remember {
|
||||||
object : (ByteArray) -> Unit {
|
object : (ByteArray) -> Unit {
|
||||||
override fun invoke(value: ByteArray) {
|
override fun invoke(value: ByteArray) {
|
||||||
@@ -170,6 +161,11 @@ fun UpdateHearingTestScreen(@Suppress("unused") navController: NavController) {
|
|||||||
leftEQ.value = parsed.leftEQ.copyOf()
|
leftEQ.value = parsed.leftEQ.copyOf()
|
||||||
rightEQ.value = parsed.rightEQ.copyOf()
|
rightEQ.value = parsed.rightEQ.copyOf()
|
||||||
conversationBoostEnabled.value = parsed.leftConversationBoost
|
conversationBoostEnabled.value = parsed.leftConversationBoost
|
||||||
|
tone.value = parsed.leftTone
|
||||||
|
ambientNoiseReduction.value = parsed.leftAmbientNoiseReduction
|
||||||
|
ownVoiceAmplification.value = parsed.ownVoiceAmplification
|
||||||
|
leftAmplification.value = parsed.leftAmplification
|
||||||
|
rightAmplification.value = parsed.rightAmplification
|
||||||
Log.d(TAG, "Updated hearing aid settings from notification")
|
Log.d(TAG, "Updated hearing aid settings from notification")
|
||||||
} else {
|
} else {
|
||||||
Log.w(TAG, "Failed to parse hearing aid settings from notification")
|
Log.w(TAG, "Failed to parse hearing aid settings from notification")
|
||||||
@@ -178,20 +174,14 @@ fun UpdateHearingTestScreen(@Suppress("unused") navController: NavController) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
LaunchedEffect(Unit) {
|
|
||||||
aacpManager?.registerControlCommandListener(AACPManager.Companion.ControlCommandIdentifiers.HEARING_AID, hearingAidListener)
|
|
||||||
aacpManager?.registerControlCommandListener(AACPManager.Companion.ControlCommandIdentifiers.HEARING_ASSIST_CONFIG, hearingAidListener)
|
|
||||||
}
|
|
||||||
|
|
||||||
DisposableEffect(Unit) {
|
DisposableEffect(Unit) {
|
||||||
onDispose {
|
onDispose {
|
||||||
aacpManager?.unregisterControlCommandListener(AACPManager.Companion.ControlCommandIdentifiers.HEARING_AID, hearingAidListener)
|
|
||||||
aacpManager?.unregisterControlCommandListener(AACPManager.Companion.ControlCommandIdentifiers.HEARING_ASSIST_CONFIG, hearingAidListener)
|
|
||||||
attManager.unregisterListener(ATTHandles.HEARING_AID, hearingAidATTListener)
|
attManager.unregisterListener(ATTHandles.HEARING_AID, hearingAidATTListener)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
LaunchedEffect(leftEQ.value, rightEQ.value, conversationBoostEnabled.value, initialLoadComplete.value, initialReadSucceeded.value) {
|
LaunchedEffect(leftEQ.value, rightEQ.value, conversationBoostEnabled.value, initialLoadComplete.value, initialReadSucceeded.value, leftAmplification.value, rightAmplification.value, tone.value, ambientNoiseReduction.value, ownVoiceAmplification.value) {
|
||||||
if (!initialLoadComplete.value) {
|
if (!initialLoadComplete.value) {
|
||||||
Log.d(TAG, "Initial device load not complete - skipping send")
|
Log.d(TAG, "Initial device load not complete - skipping send")
|
||||||
return@LaunchedEffect
|
return@LaunchedEffect
|
||||||
@@ -205,17 +195,17 @@ fun UpdateHearingTestScreen(@Suppress("unused") navController: NavController) {
|
|||||||
hearingAidSettings.value = HearingAidSettings(
|
hearingAidSettings.value = HearingAidSettings(
|
||||||
leftEQ = leftEQ.value,
|
leftEQ = leftEQ.value,
|
||||||
rightEQ = rightEQ.value,
|
rightEQ = rightEQ.value,
|
||||||
leftAmplification = 0.5f,
|
leftAmplification = leftAmplification.value,
|
||||||
rightAmplification = 0.5f,
|
rightAmplification = rightAmplification.value,
|
||||||
leftTone = 0.5f,
|
leftTone = tone.value,
|
||||||
rightTone = 0.5f,
|
rightTone = tone.value,
|
||||||
leftConversationBoost = conversationBoostEnabled.value,
|
leftConversationBoost = conversationBoostEnabled.value,
|
||||||
rightConversationBoost = conversationBoostEnabled.value,
|
rightConversationBoost = conversationBoostEnabled.value,
|
||||||
leftAmbientNoiseReduction = 0.0f,
|
leftAmbientNoiseReduction = ambientNoiseReduction.value,
|
||||||
rightAmbientNoiseReduction = 0.0f,
|
rightAmbientNoiseReduction = ambientNoiseReduction.value,
|
||||||
netAmplification = 0.5f,
|
netAmplification = leftAmplification.value + rightAmplification.value / 2,
|
||||||
balance = 0.5f,
|
balance = 0.5f + (rightAmplification.value - leftAmplification.value) / 2,
|
||||||
ownVoiceAmplification = 0.5f
|
ownVoiceAmplification = ownVoiceAmplification.value
|
||||||
)
|
)
|
||||||
Log.d(TAG, "Updated settings: ${hearingAidSettings.value}")
|
Log.d(TAG, "Updated settings: ${hearingAidSettings.value}")
|
||||||
sendHearingAidSettings(attManager, hearingAidSettings.value, debounceJob)
|
sendHearingAidSettings(attManager, hearingAidSettings.value, debounceJob)
|
||||||
@@ -227,24 +217,6 @@ fun UpdateHearingTestScreen(@Suppress("unused") navController: NavController) {
|
|||||||
attManager.enableNotifications(ATTHandles.HEARING_AID)
|
attManager.enableNotifications(ATTHandles.HEARING_AID)
|
||||||
attManager.registerListener(ATTHandles.HEARING_AID, hearingAidATTListener)
|
attManager.registerListener(ATTHandles.HEARING_AID, hearingAidATTListener)
|
||||||
|
|
||||||
try {
|
|
||||||
if (aacpManager != null) {
|
|
||||||
Log.d(TAG, "Found AACPManager, reading cached EQ data")
|
|
||||||
val aacpEQ = aacpManager.eqData
|
|
||||||
if (aacpEQ.isNotEmpty()) {
|
|
||||||
leftEQ.value = aacpEQ.copyOf()
|
|
||||||
rightEQ.value = aacpEQ.copyOf()
|
|
||||||
Log.d(TAG, "Populated EQ from AACPManager: ${aacpEQ.toList()}")
|
|
||||||
} else {
|
|
||||||
Log.d(TAG, "AACPManager EQ data empty")
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
Log.d(TAG, "No AACPManager available")
|
|
||||||
}
|
|
||||||
} catch (e: Exception) {
|
|
||||||
Log.w(TAG, "Error reading EQ from AACPManager: ${e.message}")
|
|
||||||
}
|
|
||||||
|
|
||||||
var parsedSettings: HearingAidSettings? = null
|
var parsedSettings: HearingAidSettings? = null
|
||||||
for (attempt in 1..3) {
|
for (attempt in 1..3) {
|
||||||
initialReadAttempts.intValue = attempt
|
initialReadAttempts.intValue = attempt
|
||||||
@@ -268,6 +240,11 @@ fun UpdateHearingTestScreen(@Suppress("unused") navController: NavController) {
|
|||||||
leftEQ.value = parsedSettings.leftEQ.copyOf()
|
leftEQ.value = parsedSettings.leftEQ.copyOf()
|
||||||
rightEQ.value = parsedSettings.rightEQ.copyOf()
|
rightEQ.value = parsedSettings.rightEQ.copyOf()
|
||||||
conversationBoostEnabled.value = parsedSettings.leftConversationBoost
|
conversationBoostEnabled.value = parsedSettings.leftConversationBoost
|
||||||
|
tone.value = parsedSettings.leftTone
|
||||||
|
ambientNoiseReduction.value = parsedSettings.leftAmbientNoiseReduction
|
||||||
|
ownVoiceAmplification.value = parsedSettings.ownVoiceAmplification
|
||||||
|
leftAmplification.value = parsedSettings.leftAmplification
|
||||||
|
rightAmplification.value = parsedSettings.rightAmplification
|
||||||
initialReadSucceeded.value = true
|
initialReadSucceeded.value = true
|
||||||
} else {
|
} else {
|
||||||
Log.d(TAG, "Failed to read/parse initial hearing aid settings after ${initialReadAttempts.intValue} attempts")
|
Log.d(TAG, "Failed to read/parse initial hearing aid settings after ${initialReadAttempts.intValue} attempts")
|
||||||
@@ -288,17 +265,23 @@ fun UpdateHearingTestScreen(@Suppress("unused") navController: NavController) {
|
|||||||
Spacer(modifier = Modifier.width(60.dp))
|
Spacer(modifier = Modifier.width(60.dp))
|
||||||
Text(
|
Text(
|
||||||
text = stringResource(R.string.left),
|
text = stringResource(R.string.left),
|
||||||
fontSize = 18.sp,
|
|
||||||
modifier = Modifier.weight(1f),
|
modifier = Modifier.weight(1f),
|
||||||
textAlign = TextAlign.Center,
|
textAlign = TextAlign.Center,
|
||||||
fontFamily = FontFamily(Font(R.font.sf_pro))
|
style = TextStyle(
|
||||||
|
fontSize = 18.sp,
|
||||||
|
fontFamily = FontFamily(Font(R.font.sf_pro)),
|
||||||
|
color = textColor
|
||||||
|
)
|
||||||
)
|
)
|
||||||
Text(
|
Text(
|
||||||
text = stringResource(R.string.right),
|
text = stringResource(R.string.right),
|
||||||
fontSize = 18.sp,
|
|
||||||
modifier = Modifier.weight(1f),
|
modifier = Modifier.weight(1f),
|
||||||
textAlign = TextAlign.Center,
|
textAlign = TextAlign.Center,
|
||||||
fontFamily = FontFamily(Font(R.font.sf_pro))
|
style = TextStyle(
|
||||||
|
fontSize = 18.sp,
|
||||||
|
fontFamily = FontFamily(Font(R.font.sf_pro)),
|
||||||
|
color = textColor
|
||||||
|
)
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -313,8 +296,11 @@ fun UpdateHearingTestScreen(@Suppress("unused") navController: NavController) {
|
|||||||
.width(60.dp)
|
.width(60.dp)
|
||||||
.align(Alignment.CenterVertically),
|
.align(Alignment.CenterVertically),
|
||||||
textAlign = TextAlign.End,
|
textAlign = TextAlign.End,
|
||||||
fontSize = 16.sp,
|
style = TextStyle(
|
||||||
fontFamily = FontFamily(Font(R.font.sf_pro)),
|
color = textColor,
|
||||||
|
fontSize = 16.sp,
|
||||||
|
fontFamily = FontFamily(Font(R.font.sf_pro))
|
||||||
|
),
|
||||||
)
|
)
|
||||||
OutlinedTextField(
|
OutlinedTextField(
|
||||||
value = leftEQ.value[index].toString(),
|
value = leftEQ.value[index].toString(),
|
||||||
@@ -324,10 +310,11 @@ fun UpdateHearingTestScreen(@Suppress("unused") navController: NavController) {
|
|||||||
val newArray = leftEQ.value.copyOf()
|
val newArray = leftEQ.value.copyOf()
|
||||||
newArray[index] = parsed
|
newArray[index] = parsed
|
||||||
leftEQ.value = newArray
|
leftEQ.value = newArray
|
||||||
|
Log.d(TAG, "Left EQ updated at index $index to $parsed")
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
// label = { Text("Value", fontSize = 14.sp, fontFamily = FontFamily(Font(R.font.sf_pro))) },
|
// label = { Text("Value", fontSize = 14.sp, fontFamily = FontFamily(Font(R.font.sf_pro))) },
|
||||||
keyboardOptions = KeyboardOptions(keyboardType = KeyboardType.Number),
|
keyboardOptions = KeyboardOptions(keyboardType = KeyboardType.Decimal),
|
||||||
textStyle = TextStyle(
|
textStyle = TextStyle(
|
||||||
fontFamily = FontFamily(Font(R.font.sf_pro)),
|
fontFamily = FontFamily(Font(R.font.sf_pro)),
|
||||||
fontSize = 14.sp
|
fontSize = 14.sp
|
||||||
@@ -342,10 +329,11 @@ fun UpdateHearingTestScreen(@Suppress("unused") navController: NavController) {
|
|||||||
val newArray = rightEQ.value.copyOf()
|
val newArray = rightEQ.value.copyOf()
|
||||||
newArray[index] = parsed
|
newArray[index] = parsed
|
||||||
rightEQ.value = newArray
|
rightEQ.value = newArray
|
||||||
|
Log.d(TAG, "Right EQ updated at index $index to $parsed")
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
// label = { Text("Value", fontSize = 14.sp, fontFamily = FontFamily(Font(R.font.sf_pro))) },
|
// label = { Text("Value", fontSize = 14.sp, fontFamily = FontFamily(Font(R.font.sf_pro))) },
|
||||||
keyboardOptions = KeyboardOptions(keyboardType = KeyboardType.Number),
|
keyboardOptions = KeyboardOptions(keyboardType = KeyboardType.Decimal),
|
||||||
textStyle = TextStyle(
|
textStyle = TextStyle(
|
||||||
fontFamily = FontFamily(Font(R.font.sf_pro)),
|
fontFamily = FontFamily(Font(R.font.sf_pro)),
|
||||||
fontSize = 14.sp
|
fontSize = 14.sp
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.screens
|
package me.kavishdevar.librepods.screens
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods Contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@file:OptIn(ExperimentalEncodingApi::class)
|
@file:OptIn(ExperimentalEncodingApi::class)
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods Contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@file:OptIn(ExperimentalEncodingApi::class)
|
@file:OptIn(ExperimentalEncodingApi::class)
|
||||||
@file:Suppress("DEPRECATION")
|
@file:Suppress("DEPRECATION")
|
||||||
@@ -29,6 +29,7 @@ import android.app.NotificationManager
|
|||||||
import android.app.PendingIntent
|
import android.app.PendingIntent
|
||||||
import android.app.Service
|
import android.app.Service
|
||||||
import android.appwidget.AppWidgetManager
|
import android.appwidget.AppWidgetManager
|
||||||
|
import android.bluetooth.BluetoothAdapter
|
||||||
import android.bluetooth.BluetoothDevice
|
import android.bluetooth.BluetoothDevice
|
||||||
import android.bluetooth.BluetoothHeadset
|
import android.bluetooth.BluetoothHeadset
|
||||||
import android.bluetooth.BluetoothManager
|
import android.bluetooth.BluetoothManager
|
||||||
@@ -93,8 +94,8 @@ import me.kavishdevar.librepods.utils.AirPodsInstance
|
|||||||
import me.kavishdevar.librepods.utils.AirPodsModels
|
import me.kavishdevar.librepods.utils.AirPodsModels
|
||||||
import me.kavishdevar.librepods.utils.BLEManager
|
import me.kavishdevar.librepods.utils.BLEManager
|
||||||
import me.kavishdevar.librepods.utils.BluetoothConnectionManager
|
import me.kavishdevar.librepods.utils.BluetoothConnectionManager
|
||||||
import me.kavishdevar.librepods.utils.CrossDevice
|
//import me.kavishdevar.librepods.utils.CrossDevice
|
||||||
import me.kavishdevar.librepods.utils.CrossDevicePackets
|
//import me.kavishdevar.librepods.utils.CrossDevicePackets
|
||||||
import me.kavishdevar.librepods.utils.GestureDetector
|
import me.kavishdevar.librepods.utils.GestureDetector
|
||||||
import me.kavishdevar.librepods.utils.HeadTracking
|
import me.kavishdevar.librepods.utils.HeadTracking
|
||||||
import me.kavishdevar.librepods.utils.IslandType
|
import me.kavishdevar.librepods.utils.IslandType
|
||||||
@@ -167,7 +168,6 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
var headGestures: Boolean = true,
|
var headGestures: Boolean = true,
|
||||||
var disconnectWhenNotWearing: Boolean = false,
|
var disconnectWhenNotWearing: Boolean = false,
|
||||||
var conversationalAwarenessVolume: Int = 43,
|
var conversationalAwarenessVolume: Int = 43,
|
||||||
var textColor: Long = -1L,
|
|
||||||
var qsClickBehavior: String = "cycle",
|
var qsClickBehavior: String = "cycle",
|
||||||
var bleOnlyMode: Boolean = false,
|
var bleOnlyMode: Boolean = false,
|
||||||
|
|
||||||
@@ -193,7 +193,7 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
var leftLongPressAction: StemAction = StemAction.defaultActions[StemPressType.LONG_PRESS]!!,
|
var leftLongPressAction: StemAction = StemAction.defaultActions[StemPressType.LONG_PRESS]!!,
|
||||||
var rightLongPressAction: StemAction = StemAction.defaultActions[StemPressType.LONG_PRESS]!!,
|
var rightLongPressAction: StemAction = StemAction.defaultActions[StemPressType.LONG_PRESS]!!,
|
||||||
|
|
||||||
var cameraAction: AACPManager.Companion.StemPressType? = null,
|
var cameraAction: StemPressType? = null,
|
||||||
|
|
||||||
// AirPods device information
|
// AirPods device information
|
||||||
var airpodsName: String = "",
|
var airpodsName: String = "",
|
||||||
@@ -207,6 +207,9 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
var airpodsVersion3: String = "",
|
var airpodsVersion3: String = "",
|
||||||
var airpodsHardwareRevision: String = "",
|
var airpodsHardwareRevision: String = "",
|
||||||
var airpodsUpdaterIdentifier: String = "",
|
var airpodsUpdaterIdentifier: String = "",
|
||||||
|
|
||||||
|
// phone's mac, needed for tipi
|
||||||
|
var selfMacAddress: String = ""
|
||||||
)
|
)
|
||||||
|
|
||||||
private lateinit var config: ServiceConfig
|
private lateinit var config: ServiceConfig
|
||||||
@@ -250,9 +253,10 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
if (device.connectionState == "Disconnected" && !config.bleOnlyMode) {
|
if (device.connectionState == "Disconnected" && !config.bleOnlyMode) {
|
||||||
Log.d(TAG, "Seems no device has taken over, we will.")
|
Log.d(TAG, "Seems no device has taken over, we will.")
|
||||||
val bluetoothManager = getSystemService(BluetoothManager::class.java)
|
val bluetoothManager = getSystemService(BluetoothManager::class.java)
|
||||||
val bluetoothDevice = bluetoothManager.adapter.getRemoteDevice(sharedPreferences.getString(
|
val bluetoothAdapter = bluetoothManager.adapter
|
||||||
|
val bluetoothDevice = bluetoothAdapter.getRemoteDevice(sharedPreferences.getString(
|
||||||
"mac_address", "") ?: "")
|
"mac_address", "") ?: "")
|
||||||
connectToSocket(bluetoothDevice)
|
connectToSocket(bluetoothAdapter, bluetoothDevice)
|
||||||
}
|
}
|
||||||
Log.d(TAG, "Device status changed")
|
Log.d(TAG, "Device status changed")
|
||||||
if (isConnectedLocally) return
|
if (isConnectedLocally) return
|
||||||
@@ -368,9 +372,29 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
|
|
||||||
sharedPreferences.registerOnSharedPreferenceChangeListener(this)
|
sharedPreferences.registerOnSharedPreferenceChangeListener(this)
|
||||||
|
|
||||||
val process = Runtime.getRuntime().exec(arrayOf("su", "-c", "settings", "get", "secure", "bluetooth_address"))
|
localMac = config.selfMacAddress
|
||||||
val output = process.inputStream.bufferedReader().use { it.readLine() }
|
if (localMac.isEmpty()) {
|
||||||
localMac = output.trim()
|
localMac = try {
|
||||||
|
val process = Runtime.getRuntime().exec(
|
||||||
|
arrayOf("su", "-c", "settings get secure bluetooth_address")
|
||||||
|
)
|
||||||
|
|
||||||
|
val exitCode = process.waitFor()
|
||||||
|
|
||||||
|
if (exitCode == 0) {
|
||||||
|
process.inputStream.bufferedReader().use { it.readLine()?.trim().orEmpty() }
|
||||||
|
} else {
|
||||||
|
""
|
||||||
|
}
|
||||||
|
} catch (e: Exception) {
|
||||||
|
Log.e(TAG, "Error retrieving local MAC address: ${e.message}. We probably aren't rooted.")
|
||||||
|
""
|
||||||
|
}
|
||||||
|
config.selfMacAddress = localMac
|
||||||
|
sharedPreferences.edit {
|
||||||
|
putString("self_mac_address", localMac)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
ServiceManager.setService(this)
|
ServiceManager.setService(this)
|
||||||
startForegroundNotification()
|
startForegroundNotification()
|
||||||
@@ -453,8 +477,6 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
43
|
43
|
||||||
)
|
)
|
||||||
|
|
||||||
if (!contains("textColor")) putLong("textColor", -1L)
|
|
||||||
|
|
||||||
if (!contains("qs_click_behavior")) putString("qs_click_behavior", "cycle")
|
if (!contains("qs_click_behavior")) putString("qs_click_behavior", "cycle")
|
||||||
if (!contains("name")) putString("name", "AirPods")
|
if (!contains("name")) putString("name", "AirPods")
|
||||||
|
|
||||||
@@ -556,11 +578,11 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
MODE_PRIVATE
|
MODE_PRIVATE
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
Log.d(TAG, "Initializing CrossDevice")
|
// Log.d(TAG, "Initializing CrossDevice")
|
||||||
CoroutineScope(Dispatchers.IO).launch {
|
// CoroutineScope(Dispatchers.IO).launch {
|
||||||
CrossDevice.init(this@AirPodsService)
|
// CrossDevice.init(this@AirPodsService)
|
||||||
Log.d(TAG, "CrossDevice initialized")
|
// Log.d(TAG, "CrossDevice initialized")
|
||||||
}
|
// }
|
||||||
|
|
||||||
sharedPreferences = getSharedPreferences("settings", MODE_PRIVATE)
|
sharedPreferences = getSharedPreferences("settings", MODE_PRIVATE)
|
||||||
macAddress = sharedPreferences.getString("mac_address", "") ?: ""
|
macAddress = sharedPreferences.getString("mac_address", "") ?: ""
|
||||||
@@ -573,7 +595,8 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
when (state) {
|
when (state) {
|
||||||
TelephonyManager.CALL_STATE_RINGING -> {
|
TelephonyManager.CALL_STATE_RINGING -> {
|
||||||
val leAvailableForAudio = bleManager.getMostRecentStatus()?.isLeftInEar == true || bleManager.getMostRecentStatus()?.isRightInEar == true
|
val leAvailableForAudio = bleManager.getMostRecentStatus()?.isLeftInEar == true || bleManager.getMostRecentStatus()?.isRightInEar == true
|
||||||
if ((CrossDevice.isAvailable && !isConnectedLocally && earDetectionNotification.status.contains(0x00)) || leAvailableForAudio) CoroutineScope(Dispatchers.IO).launch {
|
// if ((CrossDevice.isAvailable && !isConnectedLocally && earDetectionNotification.status.contains(0x00)) || leAvailableForAudio) CoroutineScope(Dispatchers.IO).launch {
|
||||||
|
if (leAvailableForAudio) runBlocking {
|
||||||
takeOver("call")
|
takeOver("call")
|
||||||
}
|
}
|
||||||
if (config.headGestures) {
|
if (config.headGestures) {
|
||||||
@@ -583,9 +606,10 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
}
|
}
|
||||||
TelephonyManager.CALL_STATE_OFFHOOK -> {
|
TelephonyManager.CALL_STATE_OFFHOOK -> {
|
||||||
val leAvailableForAudio = bleManager.getMostRecentStatus()?.isLeftInEar == true || bleManager.getMostRecentStatus()?.isRightInEar == true
|
val leAvailableForAudio = bleManager.getMostRecentStatus()?.isLeftInEar == true || bleManager.getMostRecentStatus()?.isRightInEar == true
|
||||||
if ((CrossDevice.isAvailable && !isConnectedLocally && earDetectionNotification.status.contains(0x00)) || leAvailableForAudio) CoroutineScope(
|
// if ((CrossDevice.isAvailable && !isConnectedLocally && earDetectionNotification.status.contains(0x00)) || leAvailableForAudio) CoroutineScope(
|
||||||
|
if (leAvailableForAudio) CoroutineScope(
|
||||||
Dispatchers.IO).launch {
|
Dispatchers.IO).launch {
|
||||||
takeOver("call")
|
takeOver("call")
|
||||||
}
|
}
|
||||||
isInCall = true
|
isInCall = true
|
||||||
}
|
}
|
||||||
@@ -641,20 +665,22 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
sharedPreferences.edit { putString("name", config.deviceName) }
|
sharedPreferences.edit { putString("name", config.deviceName) }
|
||||||
}
|
}
|
||||||
|
|
||||||
Log.d("AirPodsCrossDevice", CrossDevice.isAvailable.toString())
|
// Log.d("AirPodsCrossDevice", CrossDevice.isAvailable.toString())
|
||||||
if (!CrossDevice.isAvailable) {
|
// if (!CrossDevice.isAvailable) {
|
||||||
Log.d(TAG, "${config.deviceName} connected")
|
Log.d(TAG, "${config.deviceName} connected")
|
||||||
CoroutineScope(Dispatchers.IO).launch {
|
CoroutineScope(Dispatchers.IO).launch {
|
||||||
connectToSocket(device!!)
|
val bluetoothManager = getSystemService(BluetoothManager::class.java)
|
||||||
}
|
connectToSocket(bluetoothManager.adapter, device!!)
|
||||||
Log.d(TAG, "Setting metadata")
|
|
||||||
setMetadatas(device!!)
|
|
||||||
isConnectedLocally = true
|
|
||||||
macAddress = device!!.address
|
|
||||||
sharedPreferences.edit {
|
|
||||||
putString("mac_address", macAddress)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
Log.d(TAG, "Setting metadata")
|
||||||
|
setMetadatas(device!!)
|
||||||
|
isConnectedLocally = true
|
||||||
|
macAddress = device!!.address
|
||||||
|
sharedPreferences.edit {
|
||||||
|
putString("mac_address", macAddress)
|
||||||
|
}
|
||||||
|
// }
|
||||||
|
|
||||||
} else if (intent?.action == AirPodsNotifications.AIRPODS_DISCONNECTED) {
|
} else if (intent?.action == AirPodsNotifications.AIRPODS_DISCONNECTED) {
|
||||||
device = null
|
device = null
|
||||||
isConnectedLocally = false
|
isConnectedLocally = false
|
||||||
@@ -665,7 +691,7 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
val showIslandReceiver = object: BroadcastReceiver() {
|
val showIslandReceiver = object: BroadcastReceiver() {
|
||||||
override fun onReceive(context: Context?, intent: Intent?) {
|
override fun onReceive(context: Context?, intent: Intent?) {
|
||||||
if (intent?.action == "me.kavishdevar.librepods.cross_device_island") {
|
if (intent?.action == "me.kavishdevar.librepods.cross_device_island") {
|
||||||
showIsland(this@AirPodsService, batteryNotification.getBattery().find { it.component == BatteryComponent.LEFT}?.level!!.coerceAtMost(batteryNotification.getBattery().find { it.component == BatteryComponent.RIGHT}?.level!!))
|
showIsland(this@AirPodsService, batteryNotification.getBattery().find { it.component == BatteryComponent.LEFT}?.level!!.coerceAtMost(batteryNotification.getBattery().find { it.component == BatteryComponent.RIGHT}?.level!!))
|
||||||
@@ -719,16 +745,16 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
if (profile == BluetoothProfile.A2DP) {
|
if (profile == BluetoothProfile.A2DP) {
|
||||||
val connectedDevices = proxy.connectedDevices
|
val connectedDevices = proxy.connectedDevices
|
||||||
if (connectedDevices.isNotEmpty()) {
|
if (connectedDevices.isNotEmpty()) {
|
||||||
if (!CrossDevice.isAvailable) {
|
// if (!CrossDevice.isAvailable) {
|
||||||
CoroutineScope(Dispatchers.IO).launch {
|
CoroutineScope(Dispatchers.IO).launch {
|
||||||
connectToSocket(device)
|
connectToSocket(bluetoothAdapter, device)
|
||||||
}
|
|
||||||
setMetadatas(device)
|
|
||||||
macAddress = device.address
|
|
||||||
sharedPreferences.edit {
|
|
||||||
putString("mac_address", macAddress)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
setMetadatas(device)
|
||||||
|
macAddress = device.address
|
||||||
|
sharedPreferences.edit {
|
||||||
|
putString("mac_address", macAddress)
|
||||||
|
}
|
||||||
|
// }
|
||||||
this@AirPodsService.sendBroadcast(
|
this@AirPodsService.sendBroadcast(
|
||||||
Intent(AirPodsNotifications.AIRPODS_CONNECTED)
|
Intent(AirPodsNotifications.AIRPODS_CONNECTED)
|
||||||
)
|
)
|
||||||
@@ -745,9 +771,9 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!isConnectedLocally && !CrossDevice.isAvailable) {
|
// if (!isConnectedLocally && !CrossDevice.isAvailable) {
|
||||||
clearPacketLogs()
|
// clearPacketLogs()
|
||||||
}
|
// }
|
||||||
|
|
||||||
CoroutineScope(Dispatchers.IO).launch {
|
CoroutineScope(Dispatchers.IO).launch {
|
||||||
bleManager.startScanning()
|
bleManager.startScanning()
|
||||||
@@ -819,8 +845,8 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
.getString("name", device?.name),
|
.getString("name", device?.name),
|
||||||
batteryNotification.getBattery()
|
batteryNotification.getBattery()
|
||||||
)
|
)
|
||||||
CrossDevice.sendRemotePacket(batteryInfo)
|
// CrossDevice.sendRemotePacket(batteryInfo)
|
||||||
CrossDevice.batteryBytes = batteryInfo
|
// CrossDevice.batteryBytes = batteryInfo
|
||||||
|
|
||||||
for (battery in batteryNotification.getBattery()) {
|
for (battery in batteryNotification.getBattery()) {
|
||||||
Log.d(
|
Log.d(
|
||||||
@@ -1203,7 +1229,6 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
headGestures = sharedPreferences.getBoolean("head_gestures", true),
|
headGestures = sharedPreferences.getBoolean("head_gestures", true),
|
||||||
disconnectWhenNotWearing = sharedPreferences.getBoolean("disconnect_when_not_wearing", false),
|
disconnectWhenNotWearing = sharedPreferences.getBoolean("disconnect_when_not_wearing", false),
|
||||||
conversationalAwarenessVolume = sharedPreferences.getInt("conversational_awareness_volume", 43),
|
conversationalAwarenessVolume = sharedPreferences.getInt("conversational_awareness_volume", 43),
|
||||||
textColor = sharedPreferences.getLong("textColor", -1L),
|
|
||||||
qsClickBehavior = sharedPreferences.getString("qs_click_behavior", "cycle") ?: "cycle",
|
qsClickBehavior = sharedPreferences.getString("qs_click_behavior", "cycle") ?: "cycle",
|
||||||
|
|
||||||
// AirPods state-based takeover
|
// AirPods state-based takeover
|
||||||
@@ -1229,7 +1254,7 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
leftLongPressAction = StemAction.fromString(sharedPreferences.getString("left_long_press_action", "CYCLE_NOISE_CONTROL_MODES") ?: "CYCLE_NOISE_CONTROL_MODES")!!,
|
leftLongPressAction = StemAction.fromString(sharedPreferences.getString("left_long_press_action", "CYCLE_NOISE_CONTROL_MODES") ?: "CYCLE_NOISE_CONTROL_MODES")!!,
|
||||||
rightLongPressAction = StemAction.fromString(sharedPreferences.getString("right_long_press_action", "DIGITAL_ASSISTANT") ?: "DIGITAL_ASSISTANT")!!,
|
rightLongPressAction = StemAction.fromString(sharedPreferences.getString("right_long_press_action", "DIGITAL_ASSISTANT") ?: "DIGITAL_ASSISTANT")!!,
|
||||||
|
|
||||||
cameraAction = sharedPreferences.getString("camera_action", null)?.let { AACPManager.Companion.StemPressType.valueOf(it) },
|
cameraAction = sharedPreferences.getString("camera_action", null)?.let { StemPressType.valueOf(it) },
|
||||||
|
|
||||||
// AirPods device information
|
// AirPods device information
|
||||||
airpodsName = sharedPreferences.getString("airpods_name", "") ?: "",
|
airpodsName = sharedPreferences.getString("airpods_name", "") ?: "",
|
||||||
@@ -1243,6 +1268,8 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
airpodsVersion3 = sharedPreferences.getString("airpods_version3", "") ?: "",
|
airpodsVersion3 = sharedPreferences.getString("airpods_version3", "") ?: "",
|
||||||
airpodsHardwareRevision = sharedPreferences.getString("airpods_hardware_revision", "") ?: "",
|
airpodsHardwareRevision = sharedPreferences.getString("airpods_hardware_revision", "") ?: "",
|
||||||
airpodsUpdaterIdentifier = sharedPreferences.getString("airpods_updater_identifier", "") ?: "",
|
airpodsUpdaterIdentifier = sharedPreferences.getString("airpods_updater_identifier", "") ?: "",
|
||||||
|
|
||||||
|
selfMacAddress = sharedPreferences.getString("self_mac_address", "") ?: ""
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1251,6 +1278,7 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
|
|
||||||
when(key) {
|
when(key) {
|
||||||
"name" -> config.deviceName = preferences.getString(key, "AirPods") ?: "AirPods"
|
"name" -> config.deviceName = preferences.getString(key, "AirPods") ?: "AirPods"
|
||||||
|
"mac_address" -> macAddress = preferences.getString(key, "") ?: ""
|
||||||
"automatic_ear_detection" -> config.earDetectionEnabled = preferences.getBoolean(key, true)
|
"automatic_ear_detection" -> config.earDetectionEnabled = preferences.getBoolean(key, true)
|
||||||
"conversational_awareness_pause_music" -> config.conversationalAwarenessPauseMusic = preferences.getBoolean(key, false)
|
"conversational_awareness_pause_music" -> config.conversationalAwarenessPauseMusic = preferences.getBoolean(key, false)
|
||||||
"show_phone_battery_in_widget" -> {
|
"show_phone_battery_in_widget" -> {
|
||||||
@@ -1262,7 +1290,6 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
"head_gestures" -> config.headGestures = preferences.getBoolean(key, true)
|
"head_gestures" -> config.headGestures = preferences.getBoolean(key, true)
|
||||||
"disconnect_when_not_wearing" -> config.disconnectWhenNotWearing = preferences.getBoolean(key, false)
|
"disconnect_when_not_wearing" -> config.disconnectWhenNotWearing = preferences.getBoolean(key, false)
|
||||||
"conversational_awareness_volume" -> config.conversationalAwarenessVolume = preferences.getInt(key, 43)
|
"conversational_awareness_volume" -> config.conversationalAwarenessVolume = preferences.getInt(key, 43)
|
||||||
"textColor" -> config.textColor = preferences.getLong(key, -1L)
|
|
||||||
"qs_click_behavior" -> config.qsClickBehavior = preferences.getString(key, "cycle") ?: "cycle"
|
"qs_click_behavior" -> config.qsClickBehavior = preferences.getString(key, "cycle") ?: "cycle"
|
||||||
|
|
||||||
// AirPods state-based takeover
|
// AirPods state-based takeover
|
||||||
@@ -1323,7 +1350,7 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
)!!
|
)!!
|
||||||
setupStemActions()
|
setupStemActions()
|
||||||
}
|
}
|
||||||
"camera_action" -> config.cameraAction = preferences.getString(key, null)?.let { AACPManager.Companion.StemPressType.valueOf(it) }
|
"camera_action" -> config.cameraAction = preferences.getString(key, null)?.let { StemPressType.valueOf(it) }
|
||||||
|
|
||||||
// AirPods device information
|
// AirPods device information
|
||||||
"airpods_name" -> config.airpodsName = preferences.getString(key, "") ?: ""
|
"airpods_name" -> config.airpodsName = preferences.getString(key, "") ?: ""
|
||||||
@@ -1337,10 +1364,8 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
"airpods_version3" -> config.airpodsVersion3 = preferences.getString(key, "") ?: ""
|
"airpods_version3" -> config.airpodsVersion3 = preferences.getString(key, "") ?: ""
|
||||||
"airpods_hardware_revision" -> config.airpodsHardwareRevision = preferences.getString(key, "") ?: ""
|
"airpods_hardware_revision" -> config.airpodsHardwareRevision = preferences.getString(key, "") ?: ""
|
||||||
"airpods_updater_identifier" -> config.airpodsUpdaterIdentifier = preferences.getString(key, "") ?: ""
|
"airpods_updater_identifier" -> config.airpodsUpdaterIdentifier = preferences.getString(key, "") ?: ""
|
||||||
}
|
|
||||||
|
|
||||||
if (key == "mac_address") {
|
"self_mac_address" -> config.selfMacAddress = preferences.getString(key, "") ?: ""
|
||||||
macAddress = preferences.getString(key, "") ?: ""
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1533,7 +1558,7 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
.setContentText("Unable to connect to AirPods over L2CAP")
|
.setContentText("Unable to connect to AirPods over L2CAP")
|
||||||
.setStyle(NotificationCompat.BigTextStyle()
|
.setStyle(NotificationCompat.BigTextStyle()
|
||||||
.bigText("Your AirPods are connected via Bluetooth, but LibrePods couldn't connect to AirPods using L2CAP. " +
|
.bigText("Your AirPods are connected via Bluetooth, but LibrePods couldn't connect to AirPods using L2CAP. " +
|
||||||
"Error: $errorMessage"))
|
"Error: $errorMessage"))
|
||||||
.setContentIntent(pendingIntent)
|
.setContentIntent(pendingIntent)
|
||||||
.setCategory(Notification.CATEGORY_ERROR)
|
.setCategory(Notification.CATEGORY_ERROR)
|
||||||
.setPriority(NotificationCompat.PRIORITY_HIGH)
|
.setPriority(NotificationCompat.PRIORITY_HIGH)
|
||||||
@@ -2063,56 +2088,56 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
device.METADATA_MAIN_ICON,
|
device.METADATA_MAIN_ICON,
|
||||||
resToUri(instance.model.budCaseRes).toString().toByteArray()
|
resToUri(instance.model.budCaseRes).toString().toByteArray()
|
||||||
) &&
|
) &&
|
||||||
SystemApisUtils.setMetadata(
|
SystemApisUtils.setMetadata(
|
||||||
device,
|
device,
|
||||||
device.METADATA_MODEL_NAME,
|
device.METADATA_MODEL_NAME,
|
||||||
instance.model.name.toByteArray()
|
instance.model.name.toByteArray()
|
||||||
) &&
|
) &&
|
||||||
SystemApisUtils.setMetadata(
|
SystemApisUtils.setMetadata(
|
||||||
device,
|
device,
|
||||||
device.METADATA_DEVICE_TYPE,
|
device.METADATA_DEVICE_TYPE,
|
||||||
device.DEVICE_TYPE_UNTETHERED_HEADSET.toByteArray()
|
device.DEVICE_TYPE_UNTETHERED_HEADSET.toByteArray()
|
||||||
) &&
|
) &&
|
||||||
SystemApisUtils.setMetadata(
|
SystemApisUtils.setMetadata(
|
||||||
device,
|
device,
|
||||||
device.METADATA_UNTETHERED_CASE_ICON,
|
device.METADATA_UNTETHERED_CASE_ICON,
|
||||||
resToUri(instance.model.caseRes).toString().toByteArray()
|
resToUri(instance.model.caseRes).toString().toByteArray()
|
||||||
) &&
|
) &&
|
||||||
SystemApisUtils.setMetadata(
|
SystemApisUtils.setMetadata(
|
||||||
device,
|
device,
|
||||||
device.METADATA_UNTETHERED_RIGHT_ICON,
|
device.METADATA_UNTETHERED_RIGHT_ICON,
|
||||||
resToUri(instance.model.rightBudsRes).toString().toByteArray()
|
resToUri(instance.model.rightBudsRes).toString().toByteArray()
|
||||||
) &&
|
) &&
|
||||||
SystemApisUtils.setMetadata(
|
SystemApisUtils.setMetadata(
|
||||||
device,
|
device,
|
||||||
device.METADATA_UNTETHERED_LEFT_ICON,
|
device.METADATA_UNTETHERED_LEFT_ICON,
|
||||||
resToUri(instance.model.leftBudsRes).toString().toByteArray()
|
resToUri(instance.model.leftBudsRes).toString().toByteArray()
|
||||||
) &&
|
) &&
|
||||||
SystemApisUtils.setMetadata(
|
SystemApisUtils.setMetadata(
|
||||||
device,
|
device,
|
||||||
device.METADATA_MANUFACTURER_NAME,
|
device.METADATA_MANUFACTURER_NAME,
|
||||||
instance.model.manufacturer.toByteArray()
|
instance.model.manufacturer.toByteArray()
|
||||||
) &&
|
) &&
|
||||||
SystemApisUtils.setMetadata(
|
SystemApisUtils.setMetadata(
|
||||||
device,
|
device,
|
||||||
device.METADATA_COMPANION_APP,
|
device.METADATA_COMPANION_APP,
|
||||||
"me.kavisdevar.librepods".toByteArray()
|
"me.kavishdevar.librepods".toByteArray()
|
||||||
) &&
|
) &&
|
||||||
SystemApisUtils.setMetadata(
|
SystemApisUtils.setMetadata(
|
||||||
device,
|
device,
|
||||||
device.METADATA_UNTETHERED_CASE_LOW_BATTERY_THRESHOLD,
|
device.METADATA_UNTETHERED_CASE_LOW_BATTERY_THRESHOLD,
|
||||||
"20".toByteArray()
|
"20".toByteArray()
|
||||||
) &&
|
) &&
|
||||||
SystemApisUtils.setMetadata(
|
SystemApisUtils.setMetadata(
|
||||||
device,
|
device,
|
||||||
device.METADATA_UNTETHERED_LEFT_LOW_BATTERY_THRESHOLD,
|
device.METADATA_UNTETHERED_LEFT_LOW_BATTERY_THRESHOLD,
|
||||||
"20".toByteArray()
|
"20".toByteArray()
|
||||||
) &&
|
) &&
|
||||||
SystemApisUtils.setMetadata(
|
SystemApisUtils.setMetadata(
|
||||||
device,
|
device,
|
||||||
device.METADATA_UNTETHERED_RIGHT_LOW_BATTERY_THRESHOLD,
|
device.METADATA_UNTETHERED_RIGHT_LOW_BATTERY_THRESHOLD,
|
||||||
"20".toByteArray()
|
"20".toByteArray()
|
||||||
)
|
)
|
||||||
Log.d(TAG, "Metadata set: $metadataSet")
|
Log.d(TAG, "Metadata set: $metadataSet")
|
||||||
} else {
|
} else {
|
||||||
Log.w(TAG, "AirPods instance is not of type AirPodsInstance, skipping metadata setting")
|
Log.w(TAG, "AirPods instance is not of type AirPodsInstance, skipping metadata setting")
|
||||||
@@ -2139,11 +2164,6 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
?.getString("name", bluetoothDevice?.name)
|
?.getString("name", bluetoothDevice?.name)
|
||||||
if (bluetoothDevice != null && action != null && !action.isEmpty()) {
|
if (bluetoothDevice != null && action != null && !action.isEmpty()) {
|
||||||
Log.d(TAG, "Received bluetooth connection broadcast: action=$action")
|
Log.d(TAG, "Received bluetooth connection broadcast: action=$action")
|
||||||
if (ServiceManager.getService()?.isConnectedLocally == true) {
|
|
||||||
Log.d(TAG, "Device is already connected locally, checking if we should keep audio connected")
|
|
||||||
if (ServiceManager.getService()?.socket?.isConnected == true) ServiceManager.getService()?.manuallyCheckForAudioSource() else Log.d(TAG, "We're not connected, ignoring")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if (BluetoothDevice.ACTION_ACL_CONNECTED == action) {
|
if (BluetoothDevice.ACTION_ACL_CONNECTED == action) {
|
||||||
val uuid = ParcelUuid.fromString("74ec2172-0bad-4d01-8f77-997b2be0722a")
|
val uuid = ParcelUuid.fromString("74ec2172-0bad-4d01-8f77-997b2be0722a")
|
||||||
bluetoothDevice.fetchUuidsWithSdp()
|
bluetoothDevice.fetchUuidsWithSdp()
|
||||||
@@ -2178,19 +2198,6 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
return START_STICKY
|
return START_STICKY
|
||||||
}
|
}
|
||||||
|
|
||||||
fun manuallyCheckForAudioSource() {
|
|
||||||
val shouldResume = MediaController.getMusicActive() // todo: for some reason we lose this info after disconnecting, probably android dispatches some event. haven't investigated yet.
|
|
||||||
if (airpodsInstance == null) return
|
|
||||||
Log.d(TAG, "disconnectedBecauseReversed: $disconnectedBecauseReversed, otherDeviceTookOver: $otherDeviceTookOver")
|
|
||||||
if ((earDetectionNotification.status[0] != 0.toByte() && earDetectionNotification.status[1] != 0.toByte()) || disconnectedBecauseReversed || otherDeviceTookOver) {
|
|
||||||
Log.d(
|
|
||||||
TAG,
|
|
||||||
"For some reason, Android connected to the audio profile itself even after disconnecting. Disconnecting audio profile again! I will resume: $shouldResume"
|
|
||||||
)
|
|
||||||
disconnectAudio(this, device, shouldResume = shouldResume)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@RequiresApi(Build.VERSION_CODES.R)
|
@RequiresApi(Build.VERSION_CODES.R)
|
||||||
@SuppressLint("MissingPermission", "HardwareIds")
|
@SuppressLint("MissingPermission", "HardwareIds")
|
||||||
fun takeOver(takingOverFor: String, manualTakeOverAfterReversed: Boolean = false, startHeadTrackingAgain: Boolean = false) {
|
fun takeOver(takingOverFor: String, manualTakeOverAfterReversed: Boolean = false, startHeadTrackingAgain: Boolean = false) {
|
||||||
@@ -2266,14 +2273,19 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
if (CrossDevice.isAvailable) {
|
// if (CrossDevice.isAvailable) {
|
||||||
Log.d(TAG, "CrossDevice is available, continuing")
|
// Log.d(TAG, "CrossDevice is available, continuing")
|
||||||
}
|
// }
|
||||||
else if (bleManager.getMostRecentStatus()?.isLeftInEar == true || bleManager.getMostRecentStatus()?.isRightInEar == true) {
|
// else if (bleManager.getMostRecentStatus()?.isLeftInEar == true || bleManager.getMostRecentStatus()?.isRightInEar == true) {
|
||||||
Log.d(TAG, "At least one AirPod is in ear, continuing")
|
// Log.d(TAG, "At least one AirPod is in ear, continuing")
|
||||||
}
|
// }
|
||||||
else {
|
// else {
|
||||||
Log.d(TAG, "CrossDevice not available and AirPods not in ear, skipping")
|
// Log.d(TAG, "CrossDevice not available and AirPods not in ear, skipping")
|
||||||
|
// return
|
||||||
|
// }
|
||||||
|
|
||||||
|
if (bleManager.getMostRecentStatus()?.isLeftInEar == false && bleManager.getMostRecentStatus()?.isRightInEar == false) {
|
||||||
|
Log.d(TAG, "Both AirPods are out of ear, not taking over audio")
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -2312,11 +2324,13 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
}
|
}
|
||||||
|
|
||||||
Log.d(TAG, "Taking over audio")
|
Log.d(TAG, "Taking over audio")
|
||||||
CrossDevice.sendRemotePacket(CrossDevicePackets.REQUEST_DISCONNECT.packet)
|
// CrossDevice.sendRemotePacket(CrossDevicePackets.REQUEST_DISCONNECT.packet)
|
||||||
Log.d(TAG, macAddress)
|
Log.d(TAG, macAddress)
|
||||||
|
|
||||||
sharedPreferences.edit { putBoolean("CrossDeviceIsAvailable", false) }
|
// sharedPreferences.edit { putBoolean("CrossDeviceIsAvailable", false) }
|
||||||
device = getSystemService(BluetoothManager::class.java).adapter.bondedDevices.find {
|
val bluetoothManager = getSystemService(BluetoothManager::class.java)
|
||||||
|
val bluetoothAdapter = bluetoothManager.adapter
|
||||||
|
device = bluetoothAdapter.bondedDevices.find {
|
||||||
it.address == macAddress
|
it.address == macAddress
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -2332,7 +2346,7 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
// Set a temporary connecting state
|
// Set a temporary connecting state
|
||||||
isConnectedLocally = false // Keep as false since we're not actually connecting to L2CAP
|
isConnectedLocally = false // Keep as false since we're not actually connecting to L2CAP
|
||||||
} else {
|
} else {
|
||||||
connectToSocket(device!!)
|
connectToSocket(bluetoothAdapter, device!!)
|
||||||
connectAudio(this, device)
|
connectAudio(this, device)
|
||||||
isConnectedLocally = true
|
isConnectedLocally = true
|
||||||
}
|
}
|
||||||
@@ -2340,12 +2354,13 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
showIsland(this, batteryNotification.getBattery().find { it.component == BatteryComponent.LEFT}?.level!!.coerceAtMost(batteryNotification.getBattery().find { it.component == BatteryComponent.RIGHT}?.level!!),
|
showIsland(this, batteryNotification.getBattery().find { it.component == BatteryComponent.LEFT}?.level!!.coerceAtMost(batteryNotification.getBattery().find { it.component == BatteryComponent.RIGHT}?.level!!),
|
||||||
IslandType.TAKING_OVER)
|
IslandType.TAKING_OVER)
|
||||||
|
|
||||||
CrossDevice.isAvailable = false
|
// CrossDevice.isAvailable = false
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun createBluetoothSocket(device: BluetoothDevice, uuid: ParcelUuid): BluetoothSocket {
|
private fun createBluetoothSocket(adapter: BluetoothAdapter, device: BluetoothDevice, uuid: ParcelUuid): BluetoothSocket {
|
||||||
val type = 3 // L2CAP
|
val type = 3 // L2CAP
|
||||||
val constructorSpecs = listOf(
|
val constructorSpecs = listOf(
|
||||||
|
arrayOf(adapter, device, type, true, true, 0x1001, uuid), // A16QPR3
|
||||||
arrayOf(device, type, true, true, 0x1001, uuid),
|
arrayOf(device, type, true, true, 0x1001, uuid),
|
||||||
arrayOf(device, type, 1, true, true, 0x1001, uuid),
|
arrayOf(device, type, 1, true, true, 0x1001, uuid),
|
||||||
arrayOf(type, 1, true, true, device, 0x1001, uuid),
|
arrayOf(type, 1, true, true, device, 0x1001, uuid),
|
||||||
@@ -2381,13 +2396,13 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
}
|
}
|
||||||
|
|
||||||
@SuppressLint("MissingPermission", "UnspecifiedRegisterReceiverFlag")
|
@SuppressLint("MissingPermission", "UnspecifiedRegisterReceiverFlag")
|
||||||
fun connectToSocket(device: BluetoothDevice, manual: Boolean = false) {
|
fun connectToSocket(adapter: BluetoothAdapter, device: BluetoothDevice, manual: Boolean = false) {
|
||||||
Log.d(TAG, "<LogCollector:Start> Connecting to socket")
|
Log.d(TAG, "<LogCollector:Start> Connecting to socket")
|
||||||
HiddenApiBypass.addHiddenApiExemptions("Landroid/bluetooth/BluetoothSocket;")
|
HiddenApiBypass.addHiddenApiExemptions("Landroid/bluetooth/BluetoothSocket;")
|
||||||
val uuid: ParcelUuid = ParcelUuid.fromString("74ec2172-0bad-4d01-8f77-997b2be0722a")
|
val uuid: ParcelUuid = ParcelUuid.fromString("74ec2172-0bad-4d01-8f77-997b2be0722a")
|
||||||
if (!isConnectedLocally && !CrossDevice.isAvailable) {
|
if (!isConnectedLocally) {
|
||||||
socket = try {
|
socket = try {
|
||||||
createBluetoothSocket(device, uuid)
|
createBluetoothSocket(adapter, device, uuid)
|
||||||
} catch (e: Exception) {
|
} catch (e: Exception) {
|
||||||
Log.e(TAG, "Failed to create BluetoothSocket: ${e.message}")
|
Log.e(TAG, "Failed to create BluetoothSocket: ${e.message}")
|
||||||
showSocketConnectionFailureNotification("Failed to create Bluetooth socket: ${e.localizedMessage}")
|
showSocketConnectionFailureNotification("Failed to create Bluetooth socket: ${e.localizedMessage}")
|
||||||
@@ -2404,7 +2419,7 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
|
|
||||||
BluetoothConnectionManager.setCurrentConnection(socket, device)
|
BluetoothConnectionManager.setCurrentConnection(socket, device)
|
||||||
|
|
||||||
attManager = ATTManager(device)
|
attManager = ATTManager(adapter, device)
|
||||||
attManager!!.connect()
|
attManager!!.connect()
|
||||||
|
|
||||||
// Create AirPodsInstance from stored config if available
|
// Create AirPodsInstance from stored config if available
|
||||||
@@ -2503,7 +2518,7 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
})
|
})
|
||||||
val bytes = buffer.copyOfRange(0, bytesRead)
|
val bytes = buffer.copyOfRange(0, bytesRead)
|
||||||
val formattedHex = bytes.joinToString(" ") { "%02X".format(it) }
|
val formattedHex = bytes.joinToString(" ") { "%02X".format(it) }
|
||||||
CrossDevice.sendReceivedPacket(bytes)
|
// CrossDevice.sendReceivedPacket(bytes)
|
||||||
updateNotificationContent(
|
updateNotificationContent(
|
||||||
true,
|
true,
|
||||||
sharedPreferences.getString("name", device.name),
|
sharedPreferences.getString("name", device.name),
|
||||||
@@ -2541,6 +2556,8 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
this@AirPodsService.device = device
|
this@AirPodsService.device = device
|
||||||
updateNotificationContent(false)
|
updateNotificationContent(false)
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
Log.d(TAG, "Already connected locally, skipping socket connection (isConnectedLocally = $isConnectedLocally, socket.isConnected = ${this::socket.isInitialized && socket.isConnected})")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -2566,7 +2583,7 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
override fun onServiceDisconnected(profile: Int) {}
|
override fun onServiceDisconnected(profile: Int) {}
|
||||||
}, BluetoothProfile.A2DP)
|
}, BluetoothProfile.A2DP)
|
||||||
isConnectedLocally = false
|
isConnectedLocally = false
|
||||||
CrossDevice.isAvailable = true
|
// CrossDevice.isAvailable = true
|
||||||
}
|
}
|
||||||
|
|
||||||
fun disconnectAirPods() {
|
fun disconnectAirPods() {
|
||||||
@@ -2611,20 +2628,20 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
}
|
}
|
||||||
|
|
||||||
fun getBattery(): List<Battery> {
|
fun getBattery(): List<Battery> {
|
||||||
if (!isConnectedLocally && CrossDevice.isAvailable) {
|
// if (!isConnectedLocally && CrossDevice.isAvailable) {
|
||||||
batteryNotification.setBattery(CrossDevice.batteryBytes)
|
// batteryNotification.setBattery(CrossDevice.batteryBytes)
|
||||||
}
|
// }
|
||||||
return batteryNotification.getBattery()
|
return batteryNotification.getBattery()
|
||||||
}
|
}
|
||||||
|
|
||||||
fun getANC(): Int {
|
fun getANC(): Int {
|
||||||
if (!isConnectedLocally && CrossDevice.isAvailable) {
|
// if (!isConnectedLocally && CrossDevice.isAvailable) {
|
||||||
ancNotification.setStatus(CrossDevice.ancBytes)
|
// ancNotification.setStatus(CrossDevice.ancBytes)
|
||||||
}
|
// }
|
||||||
return ancNotification.status
|
return ancNotification.status
|
||||||
}
|
}
|
||||||
|
|
||||||
fun disconnectAudio(context: Context, device: BluetoothDevice?, shouldResume: Boolean = false) {
|
fun disconnectAudio(context: Context, device: BluetoothDevice?) {
|
||||||
val bluetoothAdapter = context.getSystemService(BluetoothManager::class.java).adapter
|
val bluetoothAdapter = context.getSystemService(BluetoothManager::class.java).adapter
|
||||||
bluetoothAdapter?.getProfileProxy(context, object : BluetoothProfile.ServiceListener {
|
bluetoothAdapter?.getProfileProxy(context, object : BluetoothProfile.ServiceListener {
|
||||||
override fun onServiceConnected(profile: Int, proxy: BluetoothProfile) {
|
override fun onServiceConnected(profile: Int, proxy: BluetoothProfile) {
|
||||||
@@ -2635,13 +2652,8 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
val method =
|
val method =
|
||||||
proxy.javaClass.getMethod("disconnect", BluetoothDevice::class.java)
|
proxy.javaClass.getMethod("setConnectionPolicy", BluetoothDevice::class.java, Int::class.java)
|
||||||
method.invoke(proxy, device)
|
method.invoke(proxy, device, 0)
|
||||||
if (shouldResume) {
|
|
||||||
Handler(Looper.getMainLooper()).postDelayed({
|
|
||||||
MediaController.sendPlay()
|
|
||||||
}, 150)
|
|
||||||
}
|
|
||||||
} catch (e: Exception) {
|
} catch (e: Exception) {
|
||||||
e.printStackTrace()
|
e.printStackTrace()
|
||||||
} finally {
|
} finally {
|
||||||
@@ -2658,8 +2670,8 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
if (profile == BluetoothProfile.HEADSET) {
|
if (profile == BluetoothProfile.HEADSET) {
|
||||||
try {
|
try {
|
||||||
val method =
|
val method =
|
||||||
proxy.javaClass.getMethod("disconnect", BluetoothDevice::class.java)
|
proxy.javaClass.getMethod("setConnectionPolicy", BluetoothDevice::class.java, Int::class.java)
|
||||||
method.invoke(proxy, device)
|
method.invoke(proxy, device, 0)
|
||||||
} catch (e: Exception) {
|
} catch (e: Exception) {
|
||||||
e.printStackTrace()
|
e.printStackTrace()
|
||||||
} finally {
|
} finally {
|
||||||
@@ -2679,9 +2691,11 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
override fun onServiceConnected(profile: Int, proxy: BluetoothProfile) {
|
override fun onServiceConnected(profile: Int, proxy: BluetoothProfile) {
|
||||||
if (profile == BluetoothProfile.A2DP) {
|
if (profile == BluetoothProfile.A2DP) {
|
||||||
try {
|
try {
|
||||||
val method =
|
val policyMethod = proxy.javaClass.getMethod("setConnectionPolicy", BluetoothDevice::class.java, Int::class.java)
|
||||||
|
policyMethod.invoke(proxy, device, 100)
|
||||||
|
val connectMethod =
|
||||||
proxy.javaClass.getMethod("connect", BluetoothDevice::class.java)
|
proxy.javaClass.getMethod("connect", BluetoothDevice::class.java)
|
||||||
method.invoke(proxy, device)
|
connectMethod.invoke(proxy, device) // reduces the slight delay between allowing and actually connecting
|
||||||
} catch (e: Exception) {
|
} catch (e: Exception) {
|
||||||
e.printStackTrace()
|
e.printStackTrace()
|
||||||
} finally {
|
} finally {
|
||||||
@@ -2700,9 +2714,11 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
override fun onServiceConnected(profile: Int, proxy: BluetoothProfile) {
|
override fun onServiceConnected(profile: Int, proxy: BluetoothProfile) {
|
||||||
if (profile == BluetoothProfile.HEADSET) {
|
if (profile == BluetoothProfile.HEADSET) {
|
||||||
try {
|
try {
|
||||||
val method =
|
val policyMethod = proxy.javaClass.getMethod("setConnectionPolicy", BluetoothDevice::class.java, Int::class.java)
|
||||||
|
policyMethod.invoke(proxy, device, 100)
|
||||||
|
val connectMethod =
|
||||||
proxy.javaClass.getMethod("connect", BluetoothDevice::class.java)
|
proxy.javaClass.getMethod("connect", BluetoothDevice::class.java)
|
||||||
method.invoke(proxy, device)
|
connectMethod.invoke(proxy, device)
|
||||||
} catch (e: Exception) {
|
} catch (e: Exception) {
|
||||||
e.printStackTrace()
|
e.printStackTrace()
|
||||||
} finally {
|
} finally {
|
||||||
@@ -2761,7 +2777,7 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
}
|
}
|
||||||
telephonyManager.listen(phoneStateListener, PhoneStateListener.LISTEN_NONE)
|
telephonyManager.listen(phoneStateListener, PhoneStateListener.LISTEN_NONE)
|
||||||
isConnectedLocally = false
|
isConnectedLocally = false
|
||||||
CrossDevice.isAvailable = true
|
// CrossDevice.isAvailable = true
|
||||||
super.onDestroy()
|
super.onDestroy()
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -2802,7 +2818,7 @@ class AirPodsService : Service(), SharedPreferences.OnSharedPreferenceChangeList
|
|||||||
}
|
}
|
||||||
if (device != null) {
|
if (device != null) {
|
||||||
CoroutineScope(Dispatchers.IO).launch {
|
CoroutineScope(Dispatchers.IO).launch {
|
||||||
connectToSocket(device!!, manual = true)
|
connectToSocket(bluetoothAdapter, device!!, manual = true)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods Contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@file:OptIn(ExperimentalEncodingApi::class)
|
@file:OptIn(ExperimentalEncodingApi::class)
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|
||||||
package me.kavishdevar.librepods.ui.theme
|
package me.kavishdevar.librepods.ui.theme
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.ui.theme
|
package me.kavishdevar.librepods.ui.theme
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.ui.theme
|
package me.kavishdevar.librepods.ui.theme
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@file:OptIn(ExperimentalEncodingApi::class)
|
@file:OptIn(ExperimentalEncodingApi::class)
|
||||||
|
|
||||||
@@ -596,7 +596,7 @@ class AACPManager {
|
|||||||
eqData = FloatArray(8) { i -> eq1.get(i) }
|
eqData = FloatArray(8) { i -> eq1.get(i) }
|
||||||
Log.d(TAG, "EQ Data set to: ${eqData.toList()}, eqOnPhone: $eqOnPhone, eqOnMedia: $eqOnMedia")
|
Log.d(TAG, "EQ Data set to: ${eqData.toList()}, eqOnPhone: $eqOnPhone, eqOnMedia: $eqOnMedia")
|
||||||
}
|
}
|
||||||
|
|
||||||
Opcodes.INFORMATION -> {
|
Opcodes.INFORMATION -> {
|
||||||
Log.e(TAG, "Parsing Information Packet")
|
Log.e(TAG, "Parsing Information Packet")
|
||||||
val information = parseInformationPacket(packet)
|
val information = parseInformationPacket(packet)
|
||||||
@@ -1201,7 +1201,8 @@ class AACPManager {
|
|||||||
var offset = 9
|
var offset = 9
|
||||||
for (i in 0 until deviceCount) {
|
for (i in 0 until deviceCount) {
|
||||||
if (offset + 8 > data.size) {
|
if (offset + 8 > data.size) {
|
||||||
throw IllegalArgumentException("Data array too short to parse all connected devices")
|
Log.w(TAG, "Data array too short to parse all connected devices, returning what we have")
|
||||||
|
break
|
||||||
}
|
}
|
||||||
val macBytes = data.sliceArray(offset until offset + 6)
|
val macBytes = data.sliceArray(offset until offset + 6)
|
||||||
val mac = macBytes.joinToString(":") { "%02X".format(it) }
|
val mac = macBytes.joinToString(":") { "%02X".format(it) }
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/* This is a very basic ATT (Attribute Protocol) implementation. I have only implemented
|
/* This is a very basic ATT (Attribute Protocol) implementation. I have only implemented
|
||||||
* what is necessary for LibrePods to function, i.e. reading and writing characteristics,
|
* what is necessary for LibrePods to function, i.e. reading and writing characteristics,
|
||||||
@@ -24,6 +24,7 @@
|
|||||||
package me.kavishdevar.librepods.utils
|
package me.kavishdevar.librepods.utils
|
||||||
|
|
||||||
import android.annotation.SuppressLint
|
import android.annotation.SuppressLint
|
||||||
|
import android.bluetooth.BluetoothAdapter
|
||||||
import android.bluetooth.BluetoothDevice
|
import android.bluetooth.BluetoothDevice
|
||||||
import android.bluetooth.BluetoothSocket
|
import android.bluetooth.BluetoothSocket
|
||||||
import android.os.ParcelUuid
|
import android.os.ParcelUuid
|
||||||
@@ -49,7 +50,7 @@ enum class ATTCCCDHandles(val value: Int) {
|
|||||||
HEARING_AID(ATTHandles.HEARING_AID.value + 1),
|
HEARING_AID(ATTHandles.HEARING_AID.value + 1),
|
||||||
}
|
}
|
||||||
|
|
||||||
class ATTManager(private val device: BluetoothDevice) {
|
class ATTManager(private val adapter: BluetoothAdapter, private val device: BluetoothDevice) {
|
||||||
companion object {
|
companion object {
|
||||||
private const val TAG = "ATTManager"
|
private const val TAG = "ATTManager"
|
||||||
|
|
||||||
@@ -72,7 +73,7 @@ class ATTManager(private val device: BluetoothDevice) {
|
|||||||
HiddenApiBypass.addHiddenApiExemptions("Landroid/bluetooth/BluetoothSocket;")
|
HiddenApiBypass.addHiddenApiExemptions("Landroid/bluetooth/BluetoothSocket;")
|
||||||
val uuid = ParcelUuid.fromString("00000000-0000-0000-0000-000000000000")
|
val uuid = ParcelUuid.fromString("00000000-0000-0000-0000-000000000000")
|
||||||
|
|
||||||
socket = createBluetoothSocket(device, uuid)
|
socket = createBluetoothSocket(adapter, device, uuid)
|
||||||
socket!!.connect()
|
socket!!.connect()
|
||||||
input = socket!!.inputStream
|
input = socket!!.inputStream
|
||||||
output = socket!!.outputStream
|
output = socket!!.outputStream
|
||||||
@@ -195,9 +196,10 @@ class ATTManager(private val device: BluetoothDevice) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun createBluetoothSocket(device: BluetoothDevice, uuid: ParcelUuid): BluetoothSocket {
|
private fun createBluetoothSocket(adapter: BluetoothAdapter, device: BluetoothDevice, uuid: ParcelUuid): BluetoothSocket {
|
||||||
val type = 3 // L2CAP
|
val type = 3 // L2CAP
|
||||||
val constructorSpecs = listOf(
|
val constructorSpecs = listOf(
|
||||||
|
arrayOf(adapter, device, type, true, 31, uuid),
|
||||||
arrayOf(device, type, true, true, 31, uuid),
|
arrayOf(device, type, true, true, 31, uuid),
|
||||||
arrayOf(device, type, 1, true, true, 31, uuid),
|
arrayOf(device, type, 1, true, true, 31, uuid),
|
||||||
arrayOf(type, 1, true, true, device, 31, uuid),
|
arrayOf(type, 1, true, true, device, 31, uuid),
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.utils
|
package me.kavishdevar.librepods.utils
|
||||||
|
|
||||||
@@ -149,7 +149,8 @@ class AirPodsPro2Lightning: AirPodsBase(
|
|||||||
Capability.HEARING_AID,
|
Capability.HEARING_AID,
|
||||||
Capability.ADAPTIVE_AUDIO,
|
Capability.ADAPTIVE_AUDIO,
|
||||||
Capability.ADAPTIVE_VOLUME,
|
Capability.ADAPTIVE_VOLUME,
|
||||||
Capability.SWIPE_FOR_VOLUME
|
Capability.SWIPE_FOR_VOLUME,
|
||||||
|
Capability.HEAD_GESTURES
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -171,7 +172,8 @@ class AirPodsPro2USBC: AirPodsBase(
|
|||||||
Capability.HEARING_AID,
|
Capability.HEARING_AID,
|
||||||
Capability.ADAPTIVE_AUDIO,
|
Capability.ADAPTIVE_AUDIO,
|
||||||
Capability.ADAPTIVE_VOLUME,
|
Capability.ADAPTIVE_VOLUME,
|
||||||
Capability.SWIPE_FOR_VOLUME
|
Capability.SWIPE_FOR_VOLUME,
|
||||||
|
Capability.HEAD_GESTURES
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -230,4 +232,4 @@ object AirPodsModels {
|
|||||||
fun getModelByModelNumber(modelNumber: String): AirPodsBase? {
|
fun getModelByModelNumber(modelNumber: String): AirPodsBase? {
|
||||||
return models.find { modelNumber in it.modelNumber }
|
return models.find { modelNumber in it.modelNumber }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple's ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods Contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.utils
|
package me.kavishdevar.librepods.utils
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple's ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods Contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.utils
|
package me.kavishdevar.librepods.utils
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.utils
|
package me.kavishdevar.librepods.utils
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@file:OptIn(ExperimentalEncodingApi::class)
|
@file:OptIn(ExperimentalEncodingApi::class)
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.utils
|
package me.kavishdevar.librepods.utils
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@file:OptIn(ExperimentalEncodingApi::class)
|
@file:OptIn(ExperimentalEncodingApi::class)
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@file:Suppress("PrivatePropertyName")
|
@file:Suppress("PrivatePropertyName")
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.utils
|
package me.kavishdevar.librepods.utils
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.utils
|
package me.kavishdevar.librepods.utils
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@file:OptIn(ExperimentalEncodingApi::class)
|
@file:OptIn(ExperimentalEncodingApi::class)
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.utils
|
package me.kavishdevar.librepods.utils
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@file:OptIn(ExperimentalEncodingApi::class)
|
@file:OptIn(ExperimentalEncodingApi::class)
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|
||||||
package me.kavishdevar.librepods.utils
|
package me.kavishdevar.librepods.utils
|
||||||
|
|||||||
@@ -1,756 +0,0 @@
|
|||||||
/*
|
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
|
||||||
*
|
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
|
||||||
* This program is free software: you can redistribute it and/or modify
|
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
|
||||||
*
|
|
||||||
* 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 Affero General Public License for more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
|
|
||||||
@file:OptIn(ExperimentalEncodingApi::class)
|
|
||||||
|
|
||||||
package me.kavishdevar.librepods.utils
|
|
||||||
|
|
||||||
import android.content.Context
|
|
||||||
import android.util.Log
|
|
||||||
import androidx.compose.runtime.NoLiveLiterals
|
|
||||||
import kotlinx.coroutines.Dispatchers
|
|
||||||
import kotlinx.coroutines.flow.MutableStateFlow
|
|
||||||
import kotlinx.coroutines.flow.StateFlow
|
|
||||||
import kotlinx.coroutines.withContext
|
|
||||||
import me.kavishdevar.librepods.services.ServiceManager
|
|
||||||
import java.io.BufferedReader
|
|
||||||
import java.io.File
|
|
||||||
import java.io.FileOutputStream
|
|
||||||
import java.io.InputStreamReader
|
|
||||||
import java.net.HttpURLConnection
|
|
||||||
import java.net.URL
|
|
||||||
import kotlin.io.encoding.ExperimentalEncodingApi
|
|
||||||
|
|
||||||
@NoLiveLiterals
|
|
||||||
class RadareOffsetFinder(context: Context) {
|
|
||||||
companion object {
|
|
||||||
private const val TAG = "RadareOffsetFinder"
|
|
||||||
private const val RADARE2_URL = "https://hc-cdn.hel1.your-objectstorage.com/s/v3/c9898243c42c0d3d1387de9a37d57ce9df77f9c9_radare2-5.9.9-android-aarch64.tar.gz"
|
|
||||||
private const val HOOK_OFFSET_PROP = "persist.librepods.hook_offset"
|
|
||||||
private const val CFG_REQ_OFFSET_PROP = "persist.librepods.cfg_req_offset"
|
|
||||||
private const val CSM_CONFIG_OFFSET_PROP = "persist.librepods.csm_config_offset"
|
|
||||||
private const val PEER_INFO_REQ_OFFSET_PROP = "persist.librepods.peer_info_req_offset"
|
|
||||||
private const val SDP_OFFSET_PROP = "persist.librepods.sdp_offset"
|
|
||||||
private const val EXTRACT_DIR = "/"
|
|
||||||
|
|
||||||
private const val RADARE2_BIN_PATH = "$EXTRACT_DIR/data/local/tmp/aln_unzip/org.radare.radare2installer/radare2/bin"
|
|
||||||
private const val RADARE2_LIB_PATH = "$EXTRACT_DIR/data/local/tmp/aln_unzip/org.radare.radare2installer/radare2/lib"
|
|
||||||
private const val BUSYBOX_PATH = "$EXTRACT_DIR/data/local/tmp/aln_unzip/busybox"
|
|
||||||
|
|
||||||
private val LIBRARY_PATHS = listOf(
|
|
||||||
"/apex/com.android.bt/lib64/libbluetooth_jni.so",
|
|
||||||
"/apex/com.android.btservices/lib64/libbluetooth_jni.so",
|
|
||||||
"/system/lib64/libbluetooth_jni.so",
|
|
||||||
"/system/lib64/libbluetooth_qti.so",
|
|
||||||
"/system_ext/lib64/libbluetooth_qti.so"
|
|
||||||
)
|
|
||||||
|
|
||||||
fun findBluetoothLibraryPath(): String? {
|
|
||||||
for (path in LIBRARY_PATHS) {
|
|
||||||
if (File(path).exists()) {
|
|
||||||
Log.d(TAG, "Found Bluetooth library at $path")
|
|
||||||
return path
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Log.e(TAG, "Could not find Bluetooth library")
|
|
||||||
return null
|
|
||||||
}
|
|
||||||
|
|
||||||
fun clearHookOffsets(): Boolean {
|
|
||||||
try {
|
|
||||||
val process = Runtime.getRuntime().exec(arrayOf(
|
|
||||||
"su", "-c",
|
|
||||||
"/system/bin/setprop $HOOK_OFFSET_PROP '' && " +
|
|
||||||
"/system/bin/setprop $CFG_REQ_OFFSET_PROP '' && " +
|
|
||||||
"/system/bin/setprop $CSM_CONFIG_OFFSET_PROP '' && " +
|
|
||||||
"/system/bin/setprop $PEER_INFO_REQ_OFFSET_PROP '' &&" +
|
|
||||||
"/system/bin/setprop $SDP_OFFSET_PROP ''"
|
|
||||||
))
|
|
||||||
val exitCode = process.waitFor()
|
|
||||||
|
|
||||||
if (exitCode == 0) {
|
|
||||||
Log.d(TAG, "Successfully cleared hook offset properties")
|
|
||||||
return true
|
|
||||||
} else {
|
|
||||||
Log.e(TAG, "Failed to clear hook offset properties, exit code: $exitCode")
|
|
||||||
}
|
|
||||||
} catch (e: Exception) {
|
|
||||||
Log.e(TAG, "Error clearing hook offset properties", e)
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
fun clearSdpOffset(): Boolean {
|
|
||||||
try {
|
|
||||||
val process = Runtime.getRuntime().exec(arrayOf(
|
|
||||||
"su", "-c", "/system/bin/setprop $SDP_OFFSET_PROP ''"
|
|
||||||
))
|
|
||||||
val exitCode = process.waitFor()
|
|
||||||
|
|
||||||
if (exitCode == 0) {
|
|
||||||
Log.d(TAG, "Successfully cleared SDP offset property")
|
|
||||||
return true
|
|
||||||
} else {
|
|
||||||
Log.e(TAG, "Failed to clear SDP offset property, exit code: $exitCode")
|
|
||||||
}
|
|
||||||
} catch (e: Exception) {
|
|
||||||
Log.e(TAG, "Error clearing SDP offset property", e)
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
fun isSdpOffsetAvailable(): Boolean {
|
|
||||||
try {
|
|
||||||
val process = Runtime.getRuntime().exec(arrayOf("/system/bin/getprop", SDP_OFFSET_PROP))
|
|
||||||
val reader = BufferedReader(InputStreamReader(process.inputStream))
|
|
||||||
val propValue = reader.readLine()
|
|
||||||
process.waitFor()
|
|
||||||
|
|
||||||
if (propValue != null && propValue.isNotEmpty()) {
|
|
||||||
Log.d(TAG, "SDP offset property exists: $propValue")
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
} catch (e: Exception) {
|
|
||||||
Log.e(TAG, "Error checking if SDP offset property exists", e)
|
|
||||||
}
|
|
||||||
|
|
||||||
Log.d(TAG, "No SDP offset available")
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private val radare2TarballFile = File(context.cacheDir, "radare2.tar.gz")
|
|
||||||
|
|
||||||
private val _progressState = MutableStateFlow<ProgressState>(ProgressState.Idle)
|
|
||||||
val progressState: StateFlow<ProgressState> = _progressState
|
|
||||||
|
|
||||||
sealed class ProgressState {
|
|
||||||
object Idle : ProgressState()
|
|
||||||
object CheckingExisting : ProgressState()
|
|
||||||
object Downloading : ProgressState()
|
|
||||||
data class DownloadProgress(val progress: Float) : ProgressState()
|
|
||||||
object Extracting : ProgressState()
|
|
||||||
object MakingExecutable : ProgressState()
|
|
||||||
object FindingOffset : ProgressState()
|
|
||||||
object SavingOffset : ProgressState()
|
|
||||||
object Cleaning : ProgressState()
|
|
||||||
data class Error(val message: String) : ProgressState()
|
|
||||||
data class Success(val offset: Long) : ProgressState()
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
fun isHookOffsetAvailable(): Boolean {
|
|
||||||
Log.d(TAG, "Setup Skipped? " + ServiceManager.getService()?.applicationContext?.getSharedPreferences("settings", Context.MODE_PRIVATE)?.getBoolean("skip_setup", false).toString())
|
|
||||||
if (ServiceManager.getService()?.applicationContext?.getSharedPreferences("settings", Context.MODE_PRIVATE)?.getBoolean("skip_setup", false) == true) {
|
|
||||||
Log.d(TAG, "Setup skipped, returning true.")
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
_progressState.value = ProgressState.CheckingExisting
|
|
||||||
try {
|
|
||||||
val process = Runtime.getRuntime().exec(arrayOf("/system/bin/getprop", HOOK_OFFSET_PROP))
|
|
||||||
val reader = BufferedReader(InputStreamReader(process.inputStream))
|
|
||||||
val propValue = reader.readLine()
|
|
||||||
process.waitFor()
|
|
||||||
|
|
||||||
if (propValue != null && propValue.isNotEmpty()) {
|
|
||||||
Log.d(TAG, "Hook offset property exists: $propValue")
|
|
||||||
_progressState.value = ProgressState.Idle
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
} catch (e: Exception) {
|
|
||||||
Log.e(TAG, "Error checking if offset property exists", e)
|
|
||||||
_progressState.value = ProgressState.Error("Failed to check if offset property exists: ${e.message}")
|
|
||||||
}
|
|
||||||
|
|
||||||
Log.d(TAG, "No hook offset available")
|
|
||||||
_progressState.value = ProgressState.Idle
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
suspend fun setupAndFindOffset(): Boolean {
|
|
||||||
val offset = findOffset()
|
|
||||||
return offset > 0
|
|
||||||
}
|
|
||||||
|
|
||||||
suspend fun findOffset(): Long = withContext(Dispatchers.IO) {
|
|
||||||
try {
|
|
||||||
_progressState.value = ProgressState.Downloading
|
|
||||||
if (!downloadRadare2TarballIfNeeded()) {
|
|
||||||
_progressState.value = ProgressState.Error("Failed to download radare2 tarball")
|
|
||||||
Log.e(TAG, "Failed to download radare2 tarball")
|
|
||||||
return@withContext 0L
|
|
||||||
}
|
|
||||||
|
|
||||||
_progressState.value = ProgressState.Extracting
|
|
||||||
if (!extractRadare2Tarball()) {
|
|
||||||
_progressState.value = ProgressState.Error("Failed to extract radare2 tarball")
|
|
||||||
Log.e(TAG, "Failed to extract radare2 tarball")
|
|
||||||
return@withContext 0L
|
|
||||||
}
|
|
||||||
|
|
||||||
_progressState.value = ProgressState.MakingExecutable
|
|
||||||
if (!makeExecutable()) {
|
|
||||||
_progressState.value = ProgressState.Error("Failed to make binaries executable")
|
|
||||||
Log.e(TAG, "Failed to make binaries executable")
|
|
||||||
return@withContext 0L
|
|
||||||
}
|
|
||||||
|
|
||||||
_progressState.value = ProgressState.FindingOffset
|
|
||||||
val offset = findFunctionOffset()
|
|
||||||
if (offset == 0L) {
|
|
||||||
_progressState.value = ProgressState.Error("Failed to find function offset")
|
|
||||||
Log.e(TAG, "Failed to find function offset")
|
|
||||||
return@withContext 0L
|
|
||||||
}
|
|
||||||
|
|
||||||
_progressState.value = ProgressState.SavingOffset
|
|
||||||
if (!saveOffset(offset)) {
|
|
||||||
_progressState.value = ProgressState.Error("Failed to save offset")
|
|
||||||
Log.e(TAG, "Failed to save offset")
|
|
||||||
return@withContext 0L
|
|
||||||
}
|
|
||||||
|
|
||||||
_progressState.value = ProgressState.Cleaning
|
|
||||||
cleanupExtractedFiles()
|
|
||||||
|
|
||||||
_progressState.value = ProgressState.Success(offset)
|
|
||||||
return@withContext offset
|
|
||||||
|
|
||||||
} catch (e: Exception) {
|
|
||||||
_progressState.value = ProgressState.Error("Error: ${e.message}")
|
|
||||||
Log.e(TAG, "Error in findOffset", e)
|
|
||||||
return@withContext 0L
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private suspend fun downloadRadare2TarballIfNeeded(): Boolean = withContext(Dispatchers.IO) {
|
|
||||||
if (radare2TarballFile.exists() && radare2TarballFile.length() > 0) {
|
|
||||||
Log.d(TAG, "Radare2 tarball already downloaded to ${radare2TarballFile.absolutePath}")
|
|
||||||
return@withContext true
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
val url = URL(RADARE2_URL)
|
|
||||||
val connection = url.openConnection() as HttpURLConnection
|
|
||||||
connection.connectTimeout = 60000
|
|
||||||
connection.readTimeout = 60000
|
|
||||||
|
|
||||||
val contentLength = connection.contentLength.toFloat()
|
|
||||||
val inputStream = connection.inputStream
|
|
||||||
val outputStream = FileOutputStream(radare2TarballFile)
|
|
||||||
|
|
||||||
val buffer = ByteArray(4096)
|
|
||||||
var bytesRead: Int
|
|
||||||
var totalBytesRead = 0L
|
|
||||||
|
|
||||||
while (inputStream.read(buffer).also { bytesRead = it } != -1) {
|
|
||||||
outputStream.write(buffer, 0, bytesRead)
|
|
||||||
totalBytesRead += bytesRead
|
|
||||||
if (contentLength > 0) {
|
|
||||||
val progress = totalBytesRead.toFloat() / contentLength
|
|
||||||
_progressState.value = ProgressState.DownloadProgress(progress)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
outputStream.close()
|
|
||||||
inputStream.close()
|
|
||||||
|
|
||||||
Log.d(TAG, "Download successful to ${radare2TarballFile.absolutePath}")
|
|
||||||
return@withContext true
|
|
||||||
} catch (e: Exception) {
|
|
||||||
Log.e(TAG, "Failed to download radare2 tarball", e)
|
|
||||||
return@withContext false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private suspend fun extractRadare2Tarball(): Boolean = withContext(Dispatchers.IO) {
|
|
||||||
try {
|
|
||||||
val isAlreadyExtracted = checkIfAlreadyExtracted()
|
|
||||||
|
|
||||||
if (isAlreadyExtracted) {
|
|
||||||
Log.d(TAG, "Radare2 files already extracted correctly, skipping extraction")
|
|
||||||
return@withContext true
|
|
||||||
}
|
|
||||||
|
|
||||||
Log.d(TAG, "Removing existing extract directory")
|
|
||||||
Runtime.getRuntime().exec(arrayOf("su", "-c", "rm -rf $EXTRACT_DIR/data/local/tmp/aln_unzip")).waitFor()
|
|
||||||
|
|
||||||
Runtime.getRuntime().exec(arrayOf("su", "-c", "mkdir -p $EXTRACT_DIR/data/local/tmp/aln_unzip")).waitFor()
|
|
||||||
|
|
||||||
Log.d(TAG, "Extracting ${radare2TarballFile.absolutePath} to $EXTRACT_DIR")
|
|
||||||
|
|
||||||
val process = Runtime.getRuntime().exec(
|
|
||||||
arrayOf("su", "-c", "tar xvf ${radare2TarballFile.absolutePath} -C $EXTRACT_DIR")
|
|
||||||
)
|
|
||||||
|
|
||||||
val reader = BufferedReader(InputStreamReader(process.inputStream))
|
|
||||||
val errorReader = BufferedReader(InputStreamReader(process.errorStream))
|
|
||||||
|
|
||||||
var line: String?
|
|
||||||
while (reader.readLine().also { line = it } != null) {
|
|
||||||
Log.d(TAG, "Extract output: $line")
|
|
||||||
}
|
|
||||||
|
|
||||||
while (errorReader.readLine().also { line = it } != null) {
|
|
||||||
Log.e(TAG, "Extract error: $line")
|
|
||||||
}
|
|
||||||
|
|
||||||
val exitCode = process.waitFor()
|
|
||||||
if (exitCode == 0) {
|
|
||||||
Log.d(TAG, "Extraction completed successfully")
|
|
||||||
return@withContext true
|
|
||||||
} else {
|
|
||||||
Log.e(TAG, "Extraction failed with exit code $exitCode")
|
|
||||||
return@withContext false
|
|
||||||
}
|
|
||||||
} catch (e: Exception) {
|
|
||||||
Log.e(TAG, "Failed to extract radare2", e)
|
|
||||||
return@withContext false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private suspend fun checkIfAlreadyExtracted(): Boolean = withContext(Dispatchers.IO) {
|
|
||||||
try {
|
|
||||||
val checkDirProcess = Runtime.getRuntime().exec(
|
|
||||||
arrayOf("su", "-c", "[ -d $EXTRACT_DIR/data/local/tmp/aln_unzip ] && echo 'exists'")
|
|
||||||
)
|
|
||||||
val dirExists = BufferedReader(InputStreamReader(checkDirProcess.inputStream)).readLine() == "exists"
|
|
||||||
checkDirProcess.waitFor()
|
|
||||||
|
|
||||||
if (!dirExists) {
|
|
||||||
Log.d(TAG, "Extract directory doesn't exist, need to extract")
|
|
||||||
return@withContext false
|
|
||||||
}
|
|
||||||
|
|
||||||
val tarProcess = Runtime.getRuntime().exec(
|
|
||||||
arrayOf("su", "-c", "tar tf ${radare2TarballFile.absolutePath}")
|
|
||||||
)
|
|
||||||
val tarFiles = BufferedReader(InputStreamReader(tarProcess.inputStream)).readLines()
|
|
||||||
.filter { it.isNotEmpty() }
|
|
||||||
.map { it.trim() }
|
|
||||||
.toSet()
|
|
||||||
tarProcess.waitFor()
|
|
||||||
|
|
||||||
if (tarFiles.isEmpty()) {
|
|
||||||
Log.e(TAG, "Failed to get file list from tarball")
|
|
||||||
return@withContext false
|
|
||||||
}
|
|
||||||
|
|
||||||
val findProcess = Runtime.getRuntime().exec(
|
|
||||||
arrayOf("su", "-c", "find $EXTRACT_DIR/data/local/tmp/aln_unzip -type f | sort")
|
|
||||||
)
|
|
||||||
val extractedFiles = BufferedReader(InputStreamReader(findProcess.inputStream)).readLines()
|
|
||||||
.filter { it.isNotEmpty() }
|
|
||||||
.map { it.trim() }
|
|
||||||
.toSet()
|
|
||||||
findProcess.waitFor()
|
|
||||||
|
|
||||||
if (extractedFiles.isEmpty()) {
|
|
||||||
Log.d(TAG, "No files found in extract directory, need to extract")
|
|
||||||
return@withContext false
|
|
||||||
}
|
|
||||||
|
|
||||||
for (tarFile in tarFiles) {
|
|
||||||
if (tarFile.endsWith("/")) continue
|
|
||||||
|
|
||||||
val filePathInExtractDir = "$EXTRACT_DIR/$tarFile"
|
|
||||||
val fileCheckProcess = Runtime.getRuntime().exec(
|
|
||||||
arrayOf("su", "-c", "[ -f $filePathInExtractDir ] && echo 'exists'")
|
|
||||||
)
|
|
||||||
val fileExists = BufferedReader(InputStreamReader(fileCheckProcess.inputStream)).readLine() == "exists"
|
|
||||||
fileCheckProcess.waitFor()
|
|
||||||
|
|
||||||
if (!fileExists) {
|
|
||||||
Log.d(TAG, "File $filePathInExtractDir from tarball missing in extract directory")
|
|
||||||
Runtime.getRuntime().exec(arrayOf("su", "-c", "rm -rf $EXTRACT_DIR/data/local/tmp/aln_unzip")).waitFor()
|
|
||||||
return@withContext false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Log.d(TAG, "All ${tarFiles.size} files from tarball exist in extract directory")
|
|
||||||
return@withContext true
|
|
||||||
} catch (e: Exception) {
|
|
||||||
Log.e(TAG, "Error checking extraction status", e)
|
|
||||||
return@withContext false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private suspend fun makeExecutable(): Boolean = withContext(Dispatchers.IO) {
|
|
||||||
try {
|
|
||||||
Log.d(TAG, "Making binaries executable in $RADARE2_BIN_PATH")
|
|
||||||
val chmod1Result = Runtime.getRuntime().exec(
|
|
||||||
arrayOf("su", "-c", "chmod -R 755 $RADARE2_BIN_PATH")
|
|
||||||
).waitFor()
|
|
||||||
|
|
||||||
Log.d(TAG, "Making binaries executable in $BUSYBOX_PATH")
|
|
||||||
|
|
||||||
val chmod2Result = Runtime.getRuntime().exec(
|
|
||||||
arrayOf("su", "-c", "chmod -R 755 $BUSYBOX_PATH")
|
|
||||||
).waitFor()
|
|
||||||
|
|
||||||
if (chmod1Result == 0 && chmod2Result == 0) {
|
|
||||||
Log.d(TAG, "Successfully made binaries executable")
|
|
||||||
return@withContext true
|
|
||||||
} else {
|
|
||||||
Log.e(TAG, "Failed to make binaries executable, exit codes: $chmod1Result, $chmod2Result")
|
|
||||||
return@withContext false
|
|
||||||
}
|
|
||||||
} catch (e: Exception) {
|
|
||||||
Log.e(TAG, "Error making binaries executable", e)
|
|
||||||
return@withContext false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private suspend fun findFunctionOffset(): Long = withContext(Dispatchers.IO) {
|
|
||||||
val libraryPath = findBluetoothLibraryPath() ?: return@withContext 0L
|
|
||||||
var offset = 0L
|
|
||||||
|
|
||||||
try {
|
|
||||||
@Suppress("LocalVariableName") val currentLD_LIBRARY_PATH = ProcessBuilder().command("su", "-c", "printenv LD_LIBRARY_PATH").start().inputStream.bufferedReader().readText().trim()
|
|
||||||
val currentPATH = ProcessBuilder().command("su", "-c", "printenv PATH").start().inputStream.bufferedReader().readText().trim()
|
|
||||||
val envSetup = """
|
|
||||||
export LD_LIBRARY_PATH="$RADARE2_LIB_PATH:$currentLD_LIBRARY_PATH"
|
|
||||||
export PATH="$BUSYBOX_PATH:$RADARE2_BIN_PATH:$currentPATH"
|
|
||||||
""".trimIndent()
|
|
||||||
|
|
||||||
val command = "$envSetup && $RADARE2_BIN_PATH/rabin2 -q -E $libraryPath | grep fcr_chk_chan"
|
|
||||||
Log.d(TAG, "Running command: $command")
|
|
||||||
|
|
||||||
val process = Runtime.getRuntime().exec(arrayOf("su", "-c", command))
|
|
||||||
|
|
||||||
val reader = BufferedReader(InputStreamReader(process.inputStream))
|
|
||||||
val errorReader = BufferedReader(InputStreamReader(process.errorStream))
|
|
||||||
|
|
||||||
var line: String?
|
|
||||||
|
|
||||||
while (reader.readLine().also { line = it } != null) {
|
|
||||||
Log.d(TAG, "rabin2 output: $line")
|
|
||||||
if (line?.contains("fcr_chk_chan") == true) {
|
|
||||||
val parts = line.split(" ")
|
|
||||||
if (parts.isNotEmpty() && parts[0].startsWith("0x")) {
|
|
||||||
offset = parts[0].substring(2).toLong(16)
|
|
||||||
Log.d(TAG, "Found offset at ${parts[0]}")
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
while (errorReader.readLine().also { line = it } != null) {
|
|
||||||
Log.d(TAG, "rabin2 error: $line")
|
|
||||||
}
|
|
||||||
|
|
||||||
val exitCode = process.waitFor()
|
|
||||||
if (exitCode != 0) {
|
|
||||||
Log.e(TAG, "rabin2 command failed with exit code $exitCode")
|
|
||||||
}
|
|
||||||
|
|
||||||
// findAndSaveL2cuProcessCfgReqOffset(libraryPath, envSetup)
|
|
||||||
// findAndSaveL2cCsmConfigOffset(libraryPath, envSetup)
|
|
||||||
// findAndSaveL2cuSendPeerInfoReqOffset(libraryPath, envSetup)
|
|
||||||
|
|
||||||
// findAndSaveSdpOffset(libraryPath, envSetup) Should not be run by default, only when user asks for it.
|
|
||||||
|
|
||||||
} catch (e: Exception) {
|
|
||||||
Log.e(TAG, "Failed to find function offset", e)
|
|
||||||
return@withContext 0L
|
|
||||||
}
|
|
||||||
|
|
||||||
if (offset == 0L) {
|
|
||||||
Log.e(TAG, "Failed to extract function offset from output, aborting")
|
|
||||||
return@withContext 0L
|
|
||||||
}
|
|
||||||
|
|
||||||
Log.d(TAG, "Successfully found offset: 0x${offset.toString(16)}")
|
|
||||||
return@withContext offset
|
|
||||||
}
|
|
||||||
|
|
||||||
private suspend fun findAndSaveL2cuProcessCfgReqOffset(libraryPath: String, envSetup: String) = withContext(Dispatchers.IO) {
|
|
||||||
try {
|
|
||||||
val command = "$envSetup && $RADARE2_BIN_PATH/rabin2 -q -E $libraryPath | grep l2cu_process_our_cfg_req"
|
|
||||||
Log.d(TAG, "Running command: $command")
|
|
||||||
|
|
||||||
val process = Runtime.getRuntime().exec(arrayOf("su", "-c", command))
|
|
||||||
val reader = BufferedReader(InputStreamReader(process.inputStream))
|
|
||||||
val errorReader = BufferedReader(InputStreamReader(process.errorStream))
|
|
||||||
|
|
||||||
var line: String?
|
|
||||||
var offset = 0L
|
|
||||||
|
|
||||||
while (reader.readLine().also { line = it } != null) {
|
|
||||||
Log.d(TAG, "rabin2 output: $line")
|
|
||||||
if (line?.contains("l2cu_process_our_cfg_req") == true) {
|
|
||||||
val parts = line.split(" ")
|
|
||||||
if (parts.isNotEmpty() && parts[0].startsWith("0x")) {
|
|
||||||
offset = parts[0].substring(2).toLong(16)
|
|
||||||
Log.d(TAG, "Found l2cu_process_our_cfg_req offset at ${parts[0]}")
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
while (errorReader.readLine().also { line = it } != null) {
|
|
||||||
Log.d(TAG, "rabin2 error: $line")
|
|
||||||
}
|
|
||||||
|
|
||||||
val exitCode = process.waitFor()
|
|
||||||
if (exitCode != 0) {
|
|
||||||
Log.e(TAG, "rabin2 command failed with exit code $exitCode")
|
|
||||||
}
|
|
||||||
|
|
||||||
if (offset > 0L) {
|
|
||||||
val hexString = "0x${offset.toString(16)}"
|
|
||||||
Runtime.getRuntime().exec(arrayOf(
|
|
||||||
"su", "-c", "/system/bin/setprop $CFG_REQ_OFFSET_PROP $hexString"
|
|
||||||
)).waitFor()
|
|
||||||
Log.d(TAG, "Saved l2cu_process_our_cfg_req offset: $hexString")
|
|
||||||
}
|
|
||||||
} catch (e: Exception) {
|
|
||||||
Log.e(TAG, "Failed to find or save l2cu_process_our_cfg_req offset", e)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private suspend fun findAndSaveL2cCsmConfigOffset(libraryPath: String, envSetup: String) = withContext(Dispatchers.IO) {
|
|
||||||
try {
|
|
||||||
val command = "$envSetup && $RADARE2_BIN_PATH/rabin2 -q -E $libraryPath | grep l2c_csm_config"
|
|
||||||
Log.d(TAG, "Running command: $command")
|
|
||||||
|
|
||||||
val process = Runtime.getRuntime().exec(arrayOf("su", "-c", command))
|
|
||||||
val reader = BufferedReader(InputStreamReader(process.inputStream))
|
|
||||||
val errorReader = BufferedReader(InputStreamReader(process.errorStream))
|
|
||||||
|
|
||||||
var line: String?
|
|
||||||
var offset = 0L
|
|
||||||
|
|
||||||
while (reader.readLine().also { line = it } != null) {
|
|
||||||
Log.d(TAG, "rabin2 output: $line")
|
|
||||||
if (line?.contains("l2c_csm_config") == true) {
|
|
||||||
val parts = line.split(" ")
|
|
||||||
if (parts.isNotEmpty() && parts[0].startsWith("0x")) {
|
|
||||||
offset = parts[0].substring(2).toLong(16)
|
|
||||||
Log.d(TAG, "Found l2c_csm_config offset at ${parts[0]}")
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
while (errorReader.readLine().also { line = it } != null) {
|
|
||||||
Log.d(TAG, "rabin2 error: $line")
|
|
||||||
}
|
|
||||||
|
|
||||||
val exitCode = process.waitFor()
|
|
||||||
if (exitCode != 0) {
|
|
||||||
Log.e(TAG, "rabin2 command failed with exit code $exitCode")
|
|
||||||
}
|
|
||||||
|
|
||||||
if (offset > 0L) {
|
|
||||||
val hexString = "0x${offset.toString(16)}"
|
|
||||||
Runtime.getRuntime().exec(arrayOf(
|
|
||||||
"su", "-c", "/system/bin/setprop $CSM_CONFIG_OFFSET_PROP $hexString"
|
|
||||||
)).waitFor()
|
|
||||||
Log.d(TAG, "Saved l2c_csm_config offset: $hexString")
|
|
||||||
}
|
|
||||||
} catch (e: Exception) {
|
|
||||||
Log.e(TAG, "Failed to find or save l2c_csm_config offset", e)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private suspend fun findAndSaveL2cuSendPeerInfoReqOffset(libraryPath: String, envSetup: String) = withContext(Dispatchers.IO) {
|
|
||||||
try {
|
|
||||||
val command = "$envSetup && $RADARE2_BIN_PATH/rabin2 -q -E $libraryPath | grep l2cu_send_peer_info_req"
|
|
||||||
Log.d(TAG, "Running command: $command")
|
|
||||||
|
|
||||||
val process = Runtime.getRuntime().exec(arrayOf("su", "-c", command))
|
|
||||||
val reader = BufferedReader(InputStreamReader(process.inputStream))
|
|
||||||
val errorReader = BufferedReader(InputStreamReader(process.errorStream))
|
|
||||||
|
|
||||||
var line: String?
|
|
||||||
var offset = 0L
|
|
||||||
|
|
||||||
while (reader.readLine().also { line = it } != null) {
|
|
||||||
Log.d(TAG, "rabin2 output: $line")
|
|
||||||
if (line?.contains("l2cu_send_peer_info_req") == true) {
|
|
||||||
val parts = line.split(" ")
|
|
||||||
if (parts.isNotEmpty() && parts[0].startsWith("0x")) {
|
|
||||||
offset = parts[0].substring(2).toLong(16)
|
|
||||||
Log.d(TAG, "Found l2cu_send_peer_info_req offset at ${parts[0]}")
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
while (errorReader.readLine().also { line = it } != null) {
|
|
||||||
Log.d(TAG, "rabin2 error: $line")
|
|
||||||
}
|
|
||||||
|
|
||||||
val exitCode = process.waitFor()
|
|
||||||
if (exitCode != 0) {
|
|
||||||
Log.e(TAG, "rabin2 command failed with exit code $exitCode")
|
|
||||||
}
|
|
||||||
|
|
||||||
if (offset > 0L) {
|
|
||||||
val hexString = "0x${offset.toString(16)}"
|
|
||||||
Runtime.getRuntime().exec(arrayOf(
|
|
||||||
"su", "-c", "/system/bin/setprop $PEER_INFO_REQ_OFFSET_PROP $hexString"
|
|
||||||
)).waitFor()
|
|
||||||
Log.d(TAG, "Saved l2cu_send_peer_info_req offset: $hexString")
|
|
||||||
}
|
|
||||||
} catch (e: Exception) {
|
|
||||||
Log.e(TAG, "Failed to find or save l2cu_send_peer_info_req offset", e)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private suspend fun findAndSaveSdpOffset(libraryPath: String, envSetup: String) = withContext(Dispatchers.IO) {
|
|
||||||
try {
|
|
||||||
val command = "$envSetup && $RADARE2_BIN_PATH/rabin2 -q -E $libraryPath | grep DmSetLocalDiRecord"
|
|
||||||
Log.d(TAG, "Running command: $command")
|
|
||||||
|
|
||||||
val process = Runtime.getRuntime().exec(arrayOf("su", "-c", command))
|
|
||||||
val reader = BufferedReader(InputStreamReader(process.inputStream))
|
|
||||||
val errorReader = BufferedReader(InputStreamReader(process.errorStream))
|
|
||||||
|
|
||||||
var line: String?
|
|
||||||
var offset = 0L
|
|
||||||
|
|
||||||
while (reader.readLine().also { line = it } != null) {
|
|
||||||
Log.d(TAG, "rabin2 output: $line")
|
|
||||||
if (line?.contains("DmSetLocalDiRecord") == true) {
|
|
||||||
val parts = line.split(" ")
|
|
||||||
if (parts.isNotEmpty() && parts[0].startsWith("0x")) {
|
|
||||||
offset = parts[0].substring(2).toLong(16)
|
|
||||||
Log.d(TAG, "Found DmSetLocalDiRecord offset at ${parts[0]}")
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
while (errorReader.readLine().also { line = it } != null) {
|
|
||||||
Log.d(TAG, "rabin2 error: $line")
|
|
||||||
}
|
|
||||||
|
|
||||||
val exitCode = process.waitFor()
|
|
||||||
if (exitCode != 0) {
|
|
||||||
Log.e(TAG, "rabin2 command failed with exit code $exitCode")
|
|
||||||
}
|
|
||||||
|
|
||||||
if (offset > 0L) {
|
|
||||||
val hexString = "0x${offset.toString(16)}"
|
|
||||||
Runtime.getRuntime().exec(arrayOf(
|
|
||||||
"su", "-c", "/system/bin/setprop $SDP_OFFSET_PROP $hexString"
|
|
||||||
)).waitFor()
|
|
||||||
Log.d(TAG, "Saved DmSetLocalDiRecord offset: $hexString")
|
|
||||||
}
|
|
||||||
} catch (e: Exception) {
|
|
||||||
Log.e(TAG, "Failed to find or save DmSetLocalDiRecord offset", e)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private suspend fun saveOffset(offset: Long): Boolean = withContext(Dispatchers.IO) {
|
|
||||||
try {
|
|
||||||
val hexString = "0x${offset.toString(16)}"
|
|
||||||
Log.d(TAG, "Saving offset to system property: $hexString")
|
|
||||||
|
|
||||||
val process = Runtime.getRuntime().exec(arrayOf(
|
|
||||||
"su", "-c", "/system/bin/setprop $HOOK_OFFSET_PROP $hexString"
|
|
||||||
))
|
|
||||||
|
|
||||||
val exitCode = process.waitFor()
|
|
||||||
if (exitCode == 0) {
|
|
||||||
val verifyProcess = Runtime.getRuntime().exec(arrayOf(
|
|
||||||
"/system/bin/getprop", HOOK_OFFSET_PROP
|
|
||||||
))
|
|
||||||
val propValue = BufferedReader(InputStreamReader(verifyProcess.inputStream)).readLine()
|
|
||||||
verifyProcess.waitFor()
|
|
||||||
|
|
||||||
if (propValue != null && propValue.isNotEmpty()) {
|
|
||||||
Log.d(TAG, "Successfully saved offset to system property: $propValue")
|
|
||||||
return@withContext true
|
|
||||||
} else {
|
|
||||||
Log.e(TAG, "Property was set but couldn't be verified")
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
Log.e(TAG, "Failed to set property, exit code: $exitCode")
|
|
||||||
}
|
|
||||||
return@withContext false
|
|
||||||
} catch (e: Exception) {
|
|
||||||
Log.e(TAG, "Failed to save offset", e)
|
|
||||||
return@withContext false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun cleanupExtractedFiles() {
|
|
||||||
try {
|
|
||||||
Runtime.getRuntime().exec(arrayOf("su", "-c", "rm -rf $EXTRACT_DIR/data/local/tmp/aln_unzip")).waitFor()
|
|
||||||
Log.d(TAG, "Cleaned up extracted files at $EXTRACT_DIR/data/local/tmp/aln_unzip")
|
|
||||||
} catch (e: Exception) {
|
|
||||||
Log.e(TAG, "Failed to cleanup extracted files", e)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
suspend fun findSdpOffset(): Boolean = withContext(Dispatchers.IO) {
|
|
||||||
try {
|
|
||||||
_progressState.value = ProgressState.Downloading
|
|
||||||
if (!downloadRadare2TarballIfNeeded()) {
|
|
||||||
_progressState.value = ProgressState.Error("Failed to download radare2 tarball")
|
|
||||||
Log.e(TAG, "Failed to download radare2 tarball")
|
|
||||||
return@withContext false
|
|
||||||
}
|
|
||||||
|
|
||||||
_progressState.value = ProgressState.Extracting
|
|
||||||
if (!extractRadare2Tarball()) {
|
|
||||||
_progressState.value = ProgressState.Error("Failed to extract radare2 tarball")
|
|
||||||
Log.e(TAG, "Failed to extract radare2 tarball")
|
|
||||||
return@withContext false
|
|
||||||
}
|
|
||||||
|
|
||||||
_progressState.value = ProgressState.MakingExecutable
|
|
||||||
if (!makeExecutable()) {
|
|
||||||
_progressState.value = ProgressState.Error("Failed to make binaries executable")
|
|
||||||
Log.e(TAG, "Failed to make binaries executable")
|
|
||||||
return@withContext false
|
|
||||||
}
|
|
||||||
|
|
||||||
_progressState.value = ProgressState.FindingOffset
|
|
||||||
val libraryPath = findBluetoothLibraryPath()
|
|
||||||
if (libraryPath == null) {
|
|
||||||
_progressState.value = ProgressState.Error("Failed to find Bluetooth library")
|
|
||||||
Log.e(TAG, "Failed to find Bluetooth library")
|
|
||||||
return@withContext false
|
|
||||||
}
|
|
||||||
|
|
||||||
@Suppress("LocalVariableName") val currentLD_LIBRARY_PATH = ProcessBuilder().command("su", "-c", "printenv LD_LIBRARY_PATH").start().inputStream.bufferedReader().readText().trim()
|
|
||||||
val currentPATH = ProcessBuilder().command("su", "-c", "printenv PATH").start().inputStream.bufferedReader().readText().trim()
|
|
||||||
val envSetup = """
|
|
||||||
export LD_LIBRARY_PATH="$RADARE2_LIB_PATH:$currentLD_LIBRARY_PATH"
|
|
||||||
export PATH="$BUSYBOX_PATH:$RADARE2_BIN_PATH:$currentPATH"
|
|
||||||
""".trimIndent()
|
|
||||||
|
|
||||||
findAndSaveSdpOffset(libraryPath, envSetup)
|
|
||||||
|
|
||||||
_progressState.value = ProgressState.Cleaning
|
|
||||||
cleanupExtractedFiles()
|
|
||||||
|
|
||||||
_progressState.value = ProgressState.Success(0L)
|
|
||||||
return@withContext true
|
|
||||||
|
|
||||||
} catch (e: Exception) {
|
|
||||||
_progressState.value = ProgressState.Error("Error: ${e.message}")
|
|
||||||
Log.e(TAG, "Error in findSdpOffset", e)
|
|
||||||
return@withContext false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package me.kavishdevar.librepods.utils
|
package me.kavishdevar.librepods.utils
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@file:OptIn(ExperimentalEncodingApi::class)
|
@file:OptIn(ExperimentalEncodingApi::class)
|
||||||
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
/*
|
/*
|
||||||
* LibrePods - AirPods liberated from Apple’s ecosystem
|
LibrePods - AirPods liberated from Apple’s ecosystem
|
||||||
*
|
Copyright (C) 2025 LibrePods contributors
|
||||||
* Copyright (C) 2025 LibrePods contributors
|
|
||||||
*
|
This program is free software: you can redistribute it and/or modify
|
||||||
* 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
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
* by the Free Software Foundation, either version 3 of the License.
|
any later version.
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
* GNU Affero General Public License for more details.
|
GNU General Public License for more details.
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@file:OptIn(ExperimentalEncodingApi::class)
|
@file:OptIn(ExperimentalEncodingApi::class)
|
||||||
|
|
||||||
|
|||||||
1
android/app/src/main/res/resources.properties
Normal file
1
android/app/src/main/res/resources.properties
Normal file
@@ -0,0 +1 @@
|
|||||||
|
unqualifiedResLocale=en
|
||||||
217
android/app/src/main/res/value-it/strings.xml
Normal file
217
android/app/src/main/res/value-it/strings.xml
Normal file
@@ -0,0 +1,217 @@
|
|||||||
|
<resources xmlns:tools="http://schemas.android.com/tools" tools:ignore="MissingTranslation">
|
||||||
|
<string name="app_name" translatable="false">LibrePods</string>
|
||||||
|
<string name="app_description">Libera i tuoi AirPods dall'ecosistema Apple.</string>
|
||||||
|
<string name="app_widget_description">Visualizza lo stato della batteria dei tuoi AirPods direttamente dalla schermata principale!</string>
|
||||||
|
<string name="accessibility">Accessibilità</string>
|
||||||
|
<string name="tone_volume">Volume Tono</string>
|
||||||
|
<string name="tone_volume_description">Regola il volume del tono degli effetti sonori riprodotti dagli AirPods.</string>
|
||||||
|
<string name="audio">Audio</string>
|
||||||
|
<string name="adaptive_audio">Audio Adattivo</string>
|
||||||
|
<string name="customize_adaptive_audio">Personalizza Audio Adattivo</string>
|
||||||
|
<string name="adaptive_audio_description">L'audio adattivo risponde dinamicamente al tuo ambiente e cancella o permette i rumori esterni. Puoi personalizzare l'Audio Adattivo per permettere più o meno rumore.</string>
|
||||||
|
<string name="buds">Auricolari</string>
|
||||||
|
<string name="case_alt">Custodia</string>
|
||||||
|
<string name="test">Test</string>
|
||||||
|
<string name="name">Nome</string>
|
||||||
|
<string name="noise_control">Modalità di Ascolto</string>
|
||||||
|
<string name="off">Spento</string>
|
||||||
|
<string name="transparency">Trasparenza</string>
|
||||||
|
<string name="adaptive">Adattivo</string>
|
||||||
|
<string name="noise_cancellation">Cancellazione del Rumore</string>
|
||||||
|
<string name="press_and_hold_airpods">Premi e Tieni Premuto sugli AirPods</string>
|
||||||
|
<string name="press_and_hold_noise_control_description">Premi e tieni premuto sullo stelo per alternare tra le modalità di ascolto selezionate.</string>
|
||||||
|
<string name="head_gestures">Gesti della Testa</string>
|
||||||
|
<string name="left">Sinistra</string>
|
||||||
|
<string name="right">Destra</string>
|
||||||
|
<string name="conversational_awareness">Consapevolezza Conversazionale</string>
|
||||||
|
<string name="conversational_awareness_description">Abbassa il volume dei contenuti multimediali e riduce il rumore di fondo quando inizi a parlare con altre persone.</string>
|
||||||
|
<string name="personalized_volume">Volume Personalizzato</string>
|
||||||
|
<string name="personalized_volume_description">Regola il volume dei contenuti multimediali in risposta al tuo ambiente.</string>
|
||||||
|
<string name="noise_cancellation_single_airpod">Cancellazione del Rumore con un Solo AirPod</string>
|
||||||
|
<string name="noise_cancellation_single_airpod_description">Consenti agli AirPods di essere messi in modalità di cancellazione del rumore quando è presente un solo AirPod nell'orecchio.</string>
|
||||||
|
<string name="volume_control">Controllo Volume</string>
|
||||||
|
<string name="volume_control_description">Regola il volume scorrendo verso l'alto o verso il basso sul sensore situato sullo stelo degli AirPods Pro.</string>
|
||||||
|
<string name="airpods_not_connected">AirPods non connessi</string>
|
||||||
|
<string name="airpods_not_connected_description">Si prega di connettere i tuoi AirPods per accedere alle impostazioni.</string>
|
||||||
|
<string name="back">Indietro</string>
|
||||||
|
<string name="app_settings">Personalizzazioni</string>
|
||||||
|
<string name="relative_conversational_awareness_volume">Volume relativo</string>
|
||||||
|
<string name="relative_conversational_awareness_volume_description">Riduce a una percentuale del volume corrente invece del volume massimo.</string>
|
||||||
|
<string name="conversational_awareness_pause_music">Metti in Pausa la Musica</string>
|
||||||
|
<string name="conversational_awareness_pause_music_description">Quando inizi a parlare, la musica verrà messa in pausa.</string>
|
||||||
|
<string name="appwidget_text">ESEMPIO</string>
|
||||||
|
<string name="add_widget">Aggiungi widget</string>
|
||||||
|
<string name="noise_control_widget_description">Controlla la Modalità di Controllo del Rumore direttamente dalla tua Schermata Principale.</string>
|
||||||
|
<string name="island_connected_text">Connesso</string>
|
||||||
|
<string name="island_connected_remote_text">Connesso a Linux</string>
|
||||||
|
<string name="island_taking_over_text">Connesso</string>
|
||||||
|
<string name="island_moved_to_remote_text">Spostato su Linux</string>
|
||||||
|
<string name="island_moved_to_other_device_text">Spostato su %1$s</string>
|
||||||
|
<string name="island_moved_to_other_device_reversed_text">Riconnetti dalla notifica</string>
|
||||||
|
<string name="head_tracking">Tracciamento della Testa</string>
|
||||||
|
<string name="head_gestures_details">Annuisci per rispondere alle chiamate e scuoti la testa per rifiutarle.</string>
|
||||||
|
<string name="general_settings_header">Generale</string>
|
||||||
|
<string name="qs_click_behavior_title">Azione del Tile Impostazioni Rapide</string>
|
||||||
|
<string name="qs_click_behavior_dialog_desc">Mostra la finestra di dialogo per il controllo del rumore al tocco.</string>
|
||||||
|
<string name="qs_click_behavior_cycle_desc">Alterna tra le modalità al tocco.</string>
|
||||||
|
<string name="developer_options_header">Sviluppatore</string>
|
||||||
|
<string name="more_settings_title">Apri le Impostazioni degli AirPods</string>
|
||||||
|
<string name="more_settings_subtitle">Gestisci le funzionalità e le preferenze degli AirPods</string>
|
||||||
|
<string name="ear_detection">Rilevamento Automatico dell'Orecchio</string>
|
||||||
|
<string name="auto_play">Riproduzione Automatica</string>
|
||||||
|
<string name="auto_pause">Pausa Automatica</string>
|
||||||
|
<string name="troubleshooting">Risoluzione dei Problemi</string>
|
||||||
|
<string name="troubleshooting_description">Raccogli i log per diagnosticare i problemi con la connessione degli AirPods</string>
|
||||||
|
<string name="collect_logs">Raccogli Log</string>
|
||||||
|
<string name="saved_logs">Log Salvati</string>
|
||||||
|
<string name="no_logs_found">Nessun log salvato trovato</string>
|
||||||
|
<string name="takeover_header">Preferenze di Connessione Automatica</string>
|
||||||
|
<string name="takeover_airpods_state">Connetti ai tuoi AirPods quando il loro stato è:</string>
|
||||||
|
<string name="takeover_disconnected">Disconnesso</string>
|
||||||
|
<string name="takeover_disconnected_desc">Gli AirPods non sono connessi a un dispositivo</string>
|
||||||
|
<string name="takeover_idle">Inattivo</string>
|
||||||
|
<string name="takeover_idle_desc">Un dispositivo è connesso ai tuoi AirPods, ma non riproduce contenuti multimediali né è in chiamata</string>
|
||||||
|
<string name="takeover_music">Riproduzione di contenuti multimediali</string>
|
||||||
|
<string name="takeover_music_desc">Un dispositivo sta riproducendo contenuti multimediali sui tuoi AirPods</string>
|
||||||
|
<string name="takeover_call">In chiamata</string>
|
||||||
|
<string name="takeover_call_desc">Un dispositivo è in chiamata con i tuoi AirPods</string>
|
||||||
|
<string name="takeover_phone_state">Connetti agli AirPods quando il tuo telefono è:</string>
|
||||||
|
<string name="takeover_ringing_call">Ricezione di una chiamata</string>
|
||||||
|
<string name="takeover_ringing_call_desc">Il tuo telefono inizia a squillare</string>
|
||||||
|
<string name="takeover_media_start">Avvio della riproduzione di contenuti multimediali</string>
|
||||||
|
<string name="takeover_media_start_desc">Il tuo telefono inizia a riprodurre contenuti multimediali</string>
|
||||||
|
<string name="undo">Annulla</string>
|
||||||
|
<string name="customize_transparency_mode_description">Puoi personalizzare la modalità Trasparenza per i tuoi AirPods Pro per aiutarti a sentire ciò che ti circonda.</string>
|
||||||
|
<string name="loud_sound_reduction_description">La Riduzione dei Suoni Forti può ridurre attivamente la tua esposizione ai forti rumori ambientali quando in modalità Trasparenza e Adattiva. La Riduzione dei Suoni Forti non è attiva in modalità Spento.</string>
|
||||||
|
<string name="loud_sound_reduction">Riduzione dei Suoni Forti</string>
|
||||||
|
<string name="call_controls">Controlli Chiamata</string>
|
||||||
|
<string name="automatically_connect">Connetti automaticamente a questo dispositivo</string>
|
||||||
|
<string name="automatically_connect_description">Quando abilitato, gli AirPods tenteranno di connettersi automaticamente a questo dispositivo. Altrimenti, tenteranno di connettersi automaticamente solo se sono stati connessi in precedenza.</string>
|
||||||
|
<string name="sleep_detection">Metti in pausa i contenuti multimediali quando ti addormenti</string>
|
||||||
|
<string name="off_listening_mode">Modalità Ascolto Disattivata</string>
|
||||||
|
<string name="off_listening_mode_description">Quando questa opzione è attiva, le modalità di ascolto degli AirPods includeranno un'opzione "Spento". I livelli di suono forti non vengono ridotti quando la modalità di ascolto è impostata su "Spento".</string>
|
||||||
|
<string name="microphone">Microfono</string>
|
||||||
|
<string name="microphone_mode">Modalità Microfono</string>
|
||||||
|
<string name="microphone_automatic">Automatico</string>
|
||||||
|
<string name="microphone_always_right">Sempre Destro</string>
|
||||||
|
<string name="microphone_always_left">Sempre Sinistro</string>
|
||||||
|
<string name="answer_call">Rispondi alla chiamata</string>
|
||||||
|
<string name="mute_unmute">Silenzia/Riattiva</string>
|
||||||
|
<string name="hang_up">Riaggancia</string>
|
||||||
|
<string name="press_once">Premi una Volta</string>
|
||||||
|
<string name="press_twice">Premi Due Volte</string>
|
||||||
|
<string name="hearing_aid">Apparecchio Acustico</string>
|
||||||
|
<string name="adjustments">Regolazioni</string>
|
||||||
|
<string name="swipe_to_control_amplification">Scorri per controllare l'amplificazione</string>
|
||||||
|
<string name="swipe_amplification_description">Quando sei in modalità Trasparenza e nessun contenuto multimediale è in riproduzione, scorri verso l'alto e verso il basso sui controlli Touch dei tuoi AirPods Pro per aumentare o diminuire l'amplificazione dei suoni ambientali.</string>
|
||||||
|
<string name="transparency_mode">Modalità Trasparenza</string>
|
||||||
|
<string name="customize_transparency_mode">Personalizza la Modalità Trasparenza</string>
|
||||||
|
<string name="press_speed">Velocità di Pressione</string>
|
||||||
|
<string name="press_speed_description">Regola la velocità richiesta per premere due o tre volte sui tuoi AirPods.</string>
|
||||||
|
<string name="press_and_hold_duration">Durata della Pressione Prolungata</string>
|
||||||
|
<string name="press_and_hold_duration_description">Regola la durata richiesta per premere e tenere premuto sui tuoi AirPods.</string>
|
||||||
|
<string name="volume_swipe_speed">Velocità di Scorrimento del Volume</string>
|
||||||
|
<string name="volume_swipe_speed_description">Per evitare regolazioni involontarie del volume, seleziona il tempo di attesa preferito tra gli scorrimenti.</string>
|
||||||
|
<string name="equalizer">Equalizzatore</string>
|
||||||
|
<string name="apply_eq_to">Applica EQ a</string>
|
||||||
|
<string name="phone">Telefono</string>
|
||||||
|
<string name="media">Media</string>
|
||||||
|
<string name="band_label">Banda %d</string>
|
||||||
|
<string name="default_option">Predefinito</string>
|
||||||
|
<string name="slower">Più lento</string>
|
||||||
|
<string name="slowest">Il più lento</string>
|
||||||
|
<string name="longer">Più lungo</string>
|
||||||
|
<string name="longest">Il più lungo</string>
|
||||||
|
<string name="darker">Più scuro</string>
|
||||||
|
<string name="brighter">Più luminoso</string>
|
||||||
|
<string name="less">Meno</string>
|
||||||
|
<string name="more">Di più</string>
|
||||||
|
<string name="amplification">Amplificazione</string>
|
||||||
|
<string name="balance">Bilanciamento</string>
|
||||||
|
<string name="tone">Tono</string>
|
||||||
|
<string name="ambient_noise_reduction">Riduzione del Rumore Ambientale</string>
|
||||||
|
<string name="conversation_boost">Potenziamento Conversazione</string>
|
||||||
|
<string name="conversation_boost_description">Potenziamento Conversazione concentra i tuoi AirPods Pro sulla persona che parla di fronte a te, rendendo più facile sentire in una conversazione faccia a faccia.</string>
|
||||||
|
<string name="hearing_aid_description">Gli AirPods possono utilizzare i risultati di un test dell'udito per apportare modifiche che migliorano la chiarezza delle voci e dei suoni intorno a te.\n\nApparecchio Acustico è destinato solo a persone con perdita dell'udito da lieve a moderata.</string>
|
||||||
|
<string name="media_assist">Assistenza Media</string>
|
||||||
|
<string name="media_assist_description">Gli AirPods Pro possono utilizzare i risultati di un test dell'udito per apportare modifiche che migliorano la chiarezza di musica, video e chiamate.</string>
|
||||||
|
<string name="adjust_media">Regola Musica e Video</string>
|
||||||
|
<string name="adjust_calls">Regola Chiamate</string>
|
||||||
|
<string name="widget">Widget</string>
|
||||||
|
<string name="show_phone_battery_in_widget">Mostra la batteria del telefono nel widget</string>
|
||||||
|
<string name="show_phone_battery_in_widget_description">Visualizza il livello della batteria del tuo telefono nel widget accanto alla batteria degli AirPods</string>
|
||||||
|
<string name="conversational_awareness_volume">Volume Consapevolezza Conversazionale</string>
|
||||||
|
<string name="quick_settings_tile">Tile Impostazioni Rapide</string>
|
||||||
|
<string name="open_dialog_for_controlling">Apri finestra di dialogo per il controllo</string>
|
||||||
|
<string name="open_dialog_for_controlling_description">Se disabilitato, cliccando sul QS si scorrerà tra le modalità. Se abilitato, verrà mostrata una finestra di dialogo per controllare la modalità di controllo del rumore e la consapevolezza conversazionale.</string>
|
||||||
|
<string name="disconnect_when_not_wearing">Disconnetti AirPods quando non indossati</string>
|
||||||
|
<string name="disconnect_when_not_wearing_description">Sarai ancora in grado di controllarli con l'app - questo disconnette solo l'audio.</string>
|
||||||
|
<string name="advanced_options">Opzioni Avanzate</string>
|
||||||
|
<string name="set_identity_resolving_key">Imposta Chiave di Risoluzione Identità (IRK)</string>
|
||||||
|
<string name="set_identity_resolving_key_description">Imposta manualmente il valore IRK utilizzato per risolvere gli indirizzi casuali BLE</string>
|
||||||
|
<string name="set_encryption_key">Imposta Chiave di Crittografia</string>
|
||||||
|
<string name="set_encryption_key_description">Imposta manualmente il valore ENC_KEY utilizzato per decrittografare le pubblicità BLE</string>
|
||||||
|
<string name="use_alternate_head_tracking_packets">Utilizza pacchetti alternativi di tracciamento della testa</string>
|
||||||
|
<string name="use_alternate_head_tracking_packets_description">Abilita questo se il tracciamento della testa non funziona per te. Questo invia dati diversi agli AirPods per richiedere/interrompere i dati di tracciamento della testa.</string>
|
||||||
|
<string name="act_as_an_apple_device">Comportati come un dispositivo Apple</string>
|
||||||
|
<string name="act_as_an_apple_device_description">Abilita la connettività multi-dispositivo e le funzionalità di Accessibilità come la personalizzazione della modalità Trasparenza (amplificazione, tono, riduzione del rumore ambientale, potenziamento conversazione ed EQ)</string>
|
||||||
|
<string name="act_as_an_apple_device_warning">Potrebbe essere instabile!! Un massimo di due dispositivi possono essere connessi ai tuoi AirPods. Se li stai usando con un dispositivo Apple come un iPad o un Mac, connetti prima quel dispositivo e poi il tuo Android.</string>
|
||||||
|
<string name="reset_hook_offset">Reimposta Offset Hook</string>
|
||||||
|
<string name="reset_hook_offset_description">Questo cancellerà l'offset hook corrente e richiederà di rifare la procedura di configurazione. Sei sicuro di voler continuare?</string>
|
||||||
|
<string name="reset">Reimposta</string>
|
||||||
|
<string name="hook_offset_reset_success">Offset hook è stato resettato. Reindirizzamento alla configurazione...</string>
|
||||||
|
<string name="hook_offset_reset_failure">Impossibile reimpostare l'offset hook</string>
|
||||||
|
<string name="irk_set_success">IRK impostata correttamente</string>
|
||||||
|
<string name="encryption_key_set_success">Chiave di crittografia impostata correttamente</string>
|
||||||
|
<string name="irk_hex_value">Valore Esadecimale IRK</string>
|
||||||
|
<string name="enc_key_hex_value">Valore Esadecimale ENC_KEY</string>
|
||||||
|
<string name="enter_irk_hex">Inserisci IRK di 16 byte come stringa esadecimale (32 caratteri):</string>
|
||||||
|
<string name="enter_enc_key_hex">Inserisci ENC_KEY di 16 byte come stringa esadecimale (32 caratteri):</string>
|
||||||
|
<string name="must_be_32_hex_chars">Devono essere esattamente 32 caratteri esadecimali</string>
|
||||||
|
<string name="error_converting_hex">Errore durante la conversione esadecimale:</string>
|
||||||
|
<string name="found_offset_restart_bluetooth">Offset trovato, riavviare il processo Bluetooth</string>
|
||||||
|
<string name="digital_assistant">Assistente Digitale</string>
|
||||||
|
<string name="on">Attivo</string>
|
||||||
|
<string name="camera_remote">Telecomando Fotocamera</string>
|
||||||
|
<string name="camera_control">Controllo Fotocamera</string>
|
||||||
|
<string name="camera_control_description">Scatta una foto, avvia o interrompi la registrazione e altro utilizzando Premere una Volta o Premere e Tenere Premuto. Quando si utilizzano gli AirPods per le azioni della fotocamera, se si seleziona Premere una Volta, i gesti di controllo dei media non saranno disponibili e, se si seleziona Premere e Tenere Premuto, la modalità di ascolto e i gesti dell'Assistente Digitale non saranno disponibili.</string>
|
||||||
|
<string name="camera_control_app_description">Imposta un pacchetto app personalizzato per il rilevamento della fotocamera</string>
|
||||||
|
<string name="set_custom_camera_package">Imposta Appid Fotocamera Personalizzata</string>
|
||||||
|
<string name="enter_custom_camera_package">Inserisci l'id dell'applicazione della fotocamera:</string>
|
||||||
|
<string name="custom_camera_package">Appid Fotocamera Personalizzata</string>
|
||||||
|
<string name="custom_camera_package_set_success">Appid fotocamera personalizzata impostata correttamente</string>
|
||||||
|
<string name="app_listener_service_label">Ascoltatore fotocamera</string>
|
||||||
|
<string name="app_listener_service_description">Servizio di ascolto per LibrePods per rilevare quando la fotocamera è attiva per attivare il controllo della fotocamera sugli AirPods.</string>
|
||||||
|
<string name="open_source_licenses">Licenze Open Source</string>
|
||||||
|
<string name="hearing_test">Aggiorna Test Uditivo</string>
|
||||||
|
<string name="update_hearing_test">Aggiorna Risultato Test Uditivo</string>
|
||||||
|
<string name="att_manager_is_null_try_reconnecting">ATT Manager è nullo, prova a riconnetterti.</string>
|
||||||
|
<string name="permissions_required">Sono richieste le seguenti autorizzazioni per utilizzare l'app. Si prega di concederle per continuare.</string>
|
||||||
|
<string name="shake_your_head_or_nod">Scuoti la testa o annuisci!</string>
|
||||||
|
<string name="root_access_required">Accesso Root Richiesto</string>
|
||||||
|
<string name="this_app_needs_root_access_to_hook_onto_the_bluetooth_library">Questa app ha bisogno dell'accesso root per agganciarsi alla libreria Bluetooth</string>
|
||||||
|
<string name="root_access_denied">L'accesso root è stato negato. Si prega di concedere i permessi di root.</string>
|
||||||
|
<string name="troubleshooting_steps">Passaggi per la Risoluzione dei Problemi</string>
|
||||||
|
<string name="hearing_test_value_instruction">Si prega di inserire i valori di perdita in dbHL</string>
|
||||||
|
<string name="about">Informazioni</string>
|
||||||
|
<string name="model_name">Nome Modello</string>
|
||||||
|
<string name="model_number">Numero Modello</string>
|
||||||
|
<string name="serial_number">Numero di Serie</string>
|
||||||
|
<string name="version">Versione</string>
|
||||||
|
<string name="hearing_health">Salute Uditiva</string>
|
||||||
|
<string name="hearing_protection">Protezione dell'Udito</string>
|
||||||
|
<string name="workspace_use">Uso in Ambienti di Lavoro</string>
|
||||||
|
<string name="ppe">Protezione EN 352</string>
|
||||||
|
<string name="workspace_use_description">La protezione EN 352 limita il livello massimo dei media a 82 dBA e soddisfa i requisiti applicabili dello standard EN 352 per la protezione individuale dell'udito.</string>
|
||||||
|
<string name="environmental_noise">Rumore Ambientale</string>
|
||||||
|
<string name="reconnect_to_last_device">Riconnetti all'ultimo dispositivo connesso</string>
|
||||||
|
<string name="disconnect">Disconnetti</string>
|
||||||
|
<string name="support_me">Supportami</string>
|
||||||
|
<string name="never_show_again">Non mostrare più</string>
|
||||||
|
<string name="support_dialog_description">Di recente ho perso il mio AirPod sinistro. Se hai trovato utile LibrePods, considera di supportarmi su GitHub Sponsors in modo che possa acquistare un sostituto e continuare a lavorare su questo progetto: anche una piccola somma fa molto. Grazie per il tuo supporto!</string>
|
||||||
|
<string name="support_librepods">Supporta LibrePods</string>
|
||||||
|
<string name="listening_mode_off_description">Disattiva la gestione del rumore</string>
|
||||||
|
<string name="listening_mode_transparency_description">Lascia entrare i suoni esterni</string>
|
||||||
|
<string name="listening_mode_adaptive_description">Regola dinamicamente il rumore esterno</string>
|
||||||
|
<string name="listening_mode_noise_cancellation_description">Blocca i suoni esterni</string>
|
||||||
|
</resources>
|
||||||
217
android/app/src/main/res/values-es/strings.xml
Normal file
217
android/app/src/main/res/values-es/strings.xml
Normal file
@@ -0,0 +1,217 @@
|
|||||||
|
<resources xmlns:tools="http://schemas.android.com/tools" tools:ignore="MissingTranslation">
|
||||||
|
<string name="app_name" translatable="false">LibrePods</string>
|
||||||
|
<string name="app_description">Libera tus AirPods del ecosistema de Apple.</string>
|
||||||
|
<string name="app_widget_description">¡Ve el estado de batería de tus AirPods desde tu pantalla de inicio!</string>
|
||||||
|
<string name="accessibility">Accesibilidad</string>
|
||||||
|
<string name="tone_volume">Volumen del tono</string>
|
||||||
|
<string name="tone_volume_description">Ajusta el volumen de los efectos de sonido reproducidos por los AirPods.</string>
|
||||||
|
<string name="audio">Audio</string>
|
||||||
|
<string name="adaptive_audio">Audio Adaptativo</string>
|
||||||
|
<string name="customize_adaptive_audio">Personalizar Audio Adaptativo</string>
|
||||||
|
<string name="adaptive_audio_description">El audio adaptativo responde al entorno y cancela o permite ruido externo. Puedes ajustarlo para permitir más o menos ruido.</string>
|
||||||
|
<string name="buds">Auriculares</string>
|
||||||
|
<string name="case_alt">Estuche</string>
|
||||||
|
<string name="test">Probar</string>
|
||||||
|
<string name="name">Nombre</string>
|
||||||
|
<string name="noise_control">Modo de Escucha</string>
|
||||||
|
<string name="off">Desactivado</string>
|
||||||
|
<string name="transparency">Transparencia</string>
|
||||||
|
<string name="adaptive">Adaptativo</string>
|
||||||
|
<string name="noise_cancellation">Cancelación de Ruido</string>
|
||||||
|
<string name="press_and_hold_airpods">Pulsar y Mantener AirPods</string>
|
||||||
|
<string name="press_and_hold_noise_control_description">Mantén presionado para alternar entre los modos seleccionados.</string>
|
||||||
|
<string name="head_gestures">Gestos de Cabeza</string>
|
||||||
|
<string name="left">Izquierdo</string>
|
||||||
|
<string name="right">Derecho</string>
|
||||||
|
<string name="conversational_awareness">Detección de Conversación</string>
|
||||||
|
<string name="conversational_awareness_description">Reduce el volumen y el ruido de fondo cuando comienzas a hablar.</string>
|
||||||
|
<string name="personalized_volume">Volumen Personalizado</string>
|
||||||
|
<string name="personalized_volume_description">Ajusta el volumen en función del entorno.</string>
|
||||||
|
<string name="noise_cancellation_single_airpod">Cancelación de sonido con un solo AirPod</string>
|
||||||
|
<string name="noise_cancellation_single_airpod_description">Permite activar la cancelación de sonido con un solo auricular puesto.</string>
|
||||||
|
<string name="volume_control">Control de Volumen</string>
|
||||||
|
<string name="volume_control_description">Ajusta el volumen deslizando arriba o abajo en el control táctil de los AirPods Pro.</string>
|
||||||
|
<string name="airpods_not_connected">AirPods no conectados</string>
|
||||||
|
<string name="airpods_not_connected_description">Por favor, conecta tus AirPods para acceder a los ajustes.</string>
|
||||||
|
<string name="back">Atrás</string>
|
||||||
|
<string name="app_settings">Personalización</string>
|
||||||
|
<string name="relative_conversational_awareness_volume">Volumen relativo</string>
|
||||||
|
<string name="relative_conversational_awareness_volume_description">Reduce a un porcentaje del volumen actual en vez del volumen máximo.</string>
|
||||||
|
<string name="conversational_awareness_pause_music">Pausar música</string>
|
||||||
|
<string name="conversational_awareness_pause_music_description">La música se pausará cuando comiences a hablar.</string>
|
||||||
|
<string name="appwidget_text">EJEMPLO</string>
|
||||||
|
<string name="add_widget">Añadir widget</string>
|
||||||
|
<string name="noise_control_widget_description">Controla el modo de control de ruido desde tu pantalla de inicio.</string>
|
||||||
|
<string name="island_connected_text">Conectado</string>
|
||||||
|
<string name="island_connected_remote_text">Conectado a Linux</string>
|
||||||
|
<string name="island_taking_over_text">Conectado</string>
|
||||||
|
<string name="island_moved_to_remote_text">Transferido a Linux</string>
|
||||||
|
<string name="island_moved_to_other_device_text">Transferido a %1$s</string>
|
||||||
|
<string name="island_moved_to_other_device_reversed_text">Reconectar desde notificación</string>
|
||||||
|
<string name="head_tracking">Seguimiento de Cabeza</string>
|
||||||
|
<string name="head_gestures_details">Asiente para contestar y niega para rechazar.</string>
|
||||||
|
<string name="general_settings_header">General</string>
|
||||||
|
<string name="qs_click_behavior_title">Acción del botón de Ajustes Rápidos</string>
|
||||||
|
<string name="qs_click_behavior_dialog_desc">Mostrar diálogo de control de ruido al tocar.</string>
|
||||||
|
<string name="qs_click_behavior_cycle_desc">Alternar modos al tocar.</string>
|
||||||
|
<string name="developer_options_header">Desarrollador</string>
|
||||||
|
<string name="more_settings_title">Abrir Ajustes de AirPods</string>
|
||||||
|
<string name="more_settings_subtitle">Gestionar funciones y preferencias</string>
|
||||||
|
<string name="ear_detection">Detección Automática de Oído</string>
|
||||||
|
<string name="auto_play">Reproducción automática</string>
|
||||||
|
<string name="auto_pause">Pausa automática</string>
|
||||||
|
<string name="troubleshooting">Solución de Problemas</string>
|
||||||
|
<string name="troubleshooting_description">Recopila registros para diagnosticar problemas de conexión de los AirPods</string>
|
||||||
|
<string name="collect_logs">Recopilar registros</string>
|
||||||
|
<string name="saved_logs">Guardar registros</string>
|
||||||
|
<string name="no_logs_found">No se encontraron registros</string>
|
||||||
|
<string name="takeover_header">Preferencias de Autoconexión</string>
|
||||||
|
<string name="takeover_airpods_state">Conectar a tus AirPods cuando su estado sea:</string>
|
||||||
|
<string name="takeover_disconnected">Desconectado</string>
|
||||||
|
<string name="takeover_disconnected_desc">AirPods no conectados a ningún dispositivo</string>
|
||||||
|
<string name="takeover_idle">Inactivos</string>
|
||||||
|
<string name="takeover_idle_desc">Un dispositivo está conectado a tus AirPods, pero no está reproduciendo audio ni llamando</string>
|
||||||
|
<string name="takeover_music">Reproduciendo</string>
|
||||||
|
<string name="takeover_music_desc">Un dispositivo está reproduciendo audio en tus AirPods</string>
|
||||||
|
<string name="takeover_call">En llamada</string>
|
||||||
|
<string name="takeover_call_desc">Un dispositivo está en llamada con tus AirPods</string>
|
||||||
|
<string name="takeover_phone_state">Conectar a tus AirPods cuando el teléfono esté:</string>
|
||||||
|
<string name="takeover_ringing_call">Llamada entrante</string>
|
||||||
|
<string name="takeover_ringing_call_desc">El teléfono empieza a sonar</string>
|
||||||
|
<string name="takeover_media_start">Iniciando reproducción</string>
|
||||||
|
<string name="takeover_media_start_desc">El teléfono empieza a reproducir audio</string>
|
||||||
|
<string name="undo">Deshacer</string>
|
||||||
|
<string name="customize_transparency_mode_description">Puedes personalizar el modo Transparencia de tus AirPods Pro para oír mejor tu entorno.</string>
|
||||||
|
<string name="loud_sound_reduction_description">Reducción de Sonidos Fuertes puede reducir activamente la exposición a entornos ruidosos en modo Transparencia y Adaptativo. Reducción de Sonidos Fuertes no está activa en modo Desactivado.</string>
|
||||||
|
<string name="loud_sound_reduction">Reducción de Sonidos Fuertes</string>
|
||||||
|
<string name="call_controls">Controles de Llamada</string>
|
||||||
|
<string name="automatically_connect">Conectar a este dispositivo automáticamente</string>
|
||||||
|
<string name="automatically_connect_description">Al activarse, los AirPods intentarán conectarse automáticamente a este dispositivo. Si no es posible, se intentarán conectar al último dispositivo utilizado.</string>
|
||||||
|
<string name="sleep_detection">Pausar audio al quedarse dormido</string>
|
||||||
|
<string name="off_listening_mode">Modo de Escucha Desactivado</string>
|
||||||
|
<string name="off_listening_mode_description">Cuando está activado, los modos de escucha de AirPods incluyen la opción Desactivado. Cuando el modo de Escucha Desactivado está activado no se reducen ruidos fuertes.</string>
|
||||||
|
<string name="microphone">Micrófono</string>
|
||||||
|
<string name="microphone_mode">Modo de Micrófono</string>
|
||||||
|
<string name="microphone_automatic">Automático</string>
|
||||||
|
<string name="microphone_always_right">Siempre derecho</string>
|
||||||
|
<string name="microphone_always_left">Siempre izquierdo</string>
|
||||||
|
<string name="answer_call">Responder</string>
|
||||||
|
<string name="mute_unmute">Silenciar/Desilenciar</string>
|
||||||
|
<string name="hang_up">Colgar</string>
|
||||||
|
<string name="press_once">Pulsar una vez</string>
|
||||||
|
<string name="press_twice">Pulsar dos veces</string>
|
||||||
|
<string name="hearing_aid">Audífono</string>
|
||||||
|
<string name="adjustments">Ajustes</string>
|
||||||
|
<string name="swipe_to_control_amplification">Deslizar para controlar amplificación</string>
|
||||||
|
<string name="swipe_amplification_description">Cuando en modo Transparencia y no hay audio reproduciéndose, desliza hacia arriba y/o abajo en los controles táctiles de los AirPods Pro para ajustar la amplificación ambiental.</string>
|
||||||
|
<string name="transparency_mode">Modo Transparencia</string>
|
||||||
|
<string name="customize_transparency_mode">Personalizar Modo Transparencia</string>
|
||||||
|
<string name="press_speed">Velocidad de pulsación</string>
|
||||||
|
<string name="press_speed_description">Ajusta la velocidad necesaria para pulsar dos o tres veces en tus AirPods.</string>
|
||||||
|
<string name="press_and_hold_duration">Duración de pulsación prolongada</string>
|
||||||
|
<string name="press_and_hold_duration_description">Ajusta la duración requerida para pulsación prolongada en tus AirPods.</string>
|
||||||
|
<string name="volume_swipe_speed">Velocidad de deslizamiento</string>
|
||||||
|
<string name="volume_swipe_speed_description">Selecciona el tiempo entre deslizamientos para evitar ajustes involuntarios.</string>
|
||||||
|
<string name="equalizer">Ecualizador (EQ)</string>
|
||||||
|
<string name="apply_eq_to">Aplicar EQ a</string>
|
||||||
|
<string name="phone">Teléfono</string>
|
||||||
|
<string name="media">Multimedia</string>
|
||||||
|
<string name="band_label">Banda %d</string>
|
||||||
|
<string name="default_option">Predeterminado</string>
|
||||||
|
<string name="slower">Más lento</string>
|
||||||
|
<string name="slowest">Muy lento</string>
|
||||||
|
<string name="longer">Más largo</string>
|
||||||
|
<string name="longest">Muy largo</string>
|
||||||
|
<string name="darker">Más oscuro</string>
|
||||||
|
<string name="brighter">Más claro</string>
|
||||||
|
<string name="less">Menos</string>
|
||||||
|
<string name="more">Más</string>
|
||||||
|
<string name="amplification">Amplificación</string>
|
||||||
|
<string name="balance">Balance</string>
|
||||||
|
<string name="tone">Tono</string>
|
||||||
|
<string name="ambient_noise_reduction">Reducción de Ruido Ambiental</string>
|
||||||
|
<string name="conversation_boost">Refuerzo de Conversación</string>
|
||||||
|
<string name="conversation_boost_description">Refuerzo de Conversación enfoca tu AirPods Pro en la persona frente a ti facilitando la escucha de conversaciones cara a cara.</string>
|
||||||
|
<string name="hearing_aid_description">Los AirPods pueden usar resultados de pruebas auditivas para mejorar la claridad de voces y sonidos de tu alrededor.\n\nEl modo Audífono tiene como objetivo ayudar a personas con problemas auditivos leves o moderados.</string>
|
||||||
|
<string name="media_assist">Asistencia Multimedia</string>
|
||||||
|
<string name="media_assist_description">Los AirPods Pro pueden usar resultados de pruebas auditivas para mejorar la claridad de música, video y llamadas.</string>
|
||||||
|
<string name="adjust_media">Ajustar Música y Video</string>
|
||||||
|
<string name="adjust_calls">Ajustar Llamadas</string>
|
||||||
|
<string name="widget">Widget</string>
|
||||||
|
<string name="show_phone_battery_in_widget">Mostrar batería del teléfono en Widget</string>
|
||||||
|
<string name="show_phone_battery_in_widget_description">Mostrar la batería del teléfono junto a la de los AirPods en el Widget.</string>
|
||||||
|
<string name="conversational_awareness_volume">Volumen de Detección de Conversación</string>
|
||||||
|
<string name="quick_settings_tile">Botón de Ajustes Rápidos</string>
|
||||||
|
<string name="open_dialog_for_controlling">Abrir diálogo de controles</string>
|
||||||
|
<string name="open_dialog_for_controlling_description">Si está desactivado, al pulsar Ajustes rápidos alterna modos. Si está activado, muestra un diálogo de controles para control de ruido y detección de conversaciones.</string>
|
||||||
|
<string name="disconnect_when_not_wearing">Desconectar AirPods cuando no estén puestos</string>
|
||||||
|
<string name="disconnect_when_not_wearing_description">Aún podrás controlarlos con la aplicación, este ajuste sólo desconecta el audio.</string>
|
||||||
|
<string name="advanced_options">Opciones Avanzadas</string>
|
||||||
|
<string name="set_identity_resolving_key">Establecer Identity Resolving Key (IRK)</string>
|
||||||
|
<string name="set_identity_resolving_key_description">Configura manualmente valor utilizado IRK para resolver direcciones aleatorias BLE.</string>
|
||||||
|
<string name="set_encryption_key">Establecer Clave de Cifrado (ENC_KEY)</string>
|
||||||
|
<string name="set_encryption_key_description">Configurar manualmente valor ENC_KEY utilizado para descifrar BLE cifrado.</string>
|
||||||
|
<string name="use_alternate_head_tracking_packets">Utilizar paquetes head tracking alternativos</string>
|
||||||
|
<string name="use_alternate_head_tracking_packets_description">Activar si head tracking no funciona. Esto enviará datos distintos a AirPods para solicitar/detener datos head tracking.</string>
|
||||||
|
<string name="act_as_an_apple_device">Actuar como dispositivo Apple</string>
|
||||||
|
<string name="act_as_an_apple_device_description">Activa conectividad multidispositivo y funciones de Accesibilidad como modo transparencia personalizado (amplificación, tono, reducción de ruido ambiente, potenciador de conversaciones, y EQ).</string>
|
||||||
|
<string name="act_as_an_apple_device_warning">¡Puede ser inestable! Se puede conectar como máximo dos dispositivos a tus AirPods. Si estás utilizando un dispositivo de Apple, como un iPad o Mac, por favor conéctalo antes y posteriormente conecta tu dispositivo Android.</string>
|
||||||
|
<string name="reset_hook_offset">Restablecer Hook Offset</string>
|
||||||
|
<string name="reset_hook_offset_description">Esto elimina el hook offset actual y requiere volver a realizar la configuración inicial. ¿Estás seguro que quieres continuar?</string>
|
||||||
|
<string name="reset">Restablecer</string>
|
||||||
|
<string name="hook_offset_reset_success">Hook offset restablecido. Redirigiendo a configuración inicial...</string>
|
||||||
|
<string name="hook_offset_reset_failure">Error al restablecer hook offset</string>
|
||||||
|
<string name="irk_set_success">IRK ha sido establecido</string>
|
||||||
|
<string name="encryption_key_set_success">Clave de cifrado establecida</string>
|
||||||
|
<string name="irk_hex_value">Valor IRK Hex</string>
|
||||||
|
<string name="enc_key_hex_value">Valor ENC_KEY Hex</string>
|
||||||
|
<string name="enter_irk_hex">Introducir 16-byte IRK como formato hexadecimal (32 caracteres):</string>
|
||||||
|
<string name="enter_enc_key_hex">Introducir 16-byte ENC_KEY como formato hexadecimal (32 caracteres):</string>
|
||||||
|
<string name="must_be_32_hex_chars">Debe tener exactamente 32 caracteres hexadecimales</string>
|
||||||
|
<string name="error_converting_hex">Error convirtiendo hex:</string>
|
||||||
|
<string name="found_offset_restart_bluetooth">Offset encontrado. Por favor, reinicie el proceso Bluetooth</string>
|
||||||
|
<string name="digital_assistant">Asistente Digital</string>
|
||||||
|
<string name="on">Activado</string>
|
||||||
|
<string name="camera_remote">Control Remoto de Cámara</string>
|
||||||
|
<string name="camera_control">Control de Cámara</string>
|
||||||
|
<string name="camera_control_description">Toma fotos o inicia grabación usando Pulsar una vez o Pulsación Prolongada. Los AirPods para acciones de la cámara: si selecciona Pulsar una vez, los gestos de control multimedia no estarán disponibles, y si selecciona Pulsación Prolongada, el modo de escucha y los gestos del Asistente Digital no estarán disponibles.</string>
|
||||||
|
<string name="camera_control_app_description">Configurar un paquete de aplicaciones personalizado para la detección de la cámara</string>
|
||||||
|
<string name="set_custom_camera_package">Establecer Appid de cámara personalizada</string>
|
||||||
|
<string name="enter_custom_camera_package">Introduzca el ID de la aplicación de la cámara:</string>
|
||||||
|
<string name="custom_camera_package">Aplicación de cámara personalizada Appid</string>
|
||||||
|
<string name="custom_camera_package_set_success">Appid de cámara establecido correctamente</string>
|
||||||
|
<string name="app_listener_service_label">Escucha de cámara</string>
|
||||||
|
<string name="app_listener_service_description">Servicio de escucha para LibrePods que detecta cuándo la cámara está activa para activar el control de la cámara en los AirPods.</string>
|
||||||
|
<string name="open_source_licenses">Licencias de Código Abierto</string>
|
||||||
|
<string name="hearing_test">Actualizar Prueba Auditiva</string>
|
||||||
|
<string name="update_hearing_test">Actualizar el Resultado de la Prueba Auditiva</string>
|
||||||
|
<string name="att_manager_is_null_try_reconnecting">ATT Manager es nulo. Intente reconectar.</string>
|
||||||
|
<string name="permissions_required">Se requieren los siguientes permisos para utilizar la aplicación. Por favor, autorícelos para continuar.</string>
|
||||||
|
<string name="shake_your_head_or_nod">¡Mueve la cabeza o asiente!</string>
|
||||||
|
<string name="root_access_required">Se requiere acceso root</string>
|
||||||
|
<string name="this_app_needs_root_access_to_hook_onto_the_bluetooth_library">Esta aplicación necesita acceso root para conectarse a la biblioteca Bluetooth</string>
|
||||||
|
<string name="root_access_denied">Se ha denegado el acceso root. Por favor, conceda permisos root.</string>
|
||||||
|
<string name="troubleshooting_steps">Pasos para la resolución de problemas</string>
|
||||||
|
<string name="hearing_test_value_instruction">Introduzca los valores de pérdida en dbHL.</string>
|
||||||
|
<string name="about">Acerca de</string>
|
||||||
|
<string name="model_name">Nombre del modelo</string>
|
||||||
|
<string name="model_number">Número de modelo</string>
|
||||||
|
<string name="serial_number">Número de serie</string>
|
||||||
|
<string name="version">Versión</string>
|
||||||
|
<string name="hearing_health">Salud Auditiva</string>
|
||||||
|
<string name="hearing_protection">Protección Auditiva</string>
|
||||||
|
<string name="workspace_use">Workspace en uso</string>
|
||||||
|
<string name="ppe">Protección EN 352</string>
|
||||||
|
<string name="workspace_use_description">La norma EN 352 limita el nivel máximo de los medios a 82 dBA y cumple los requisitos aplicables de la norma EN 352 para la protección auditiva personal.</string>
|
||||||
|
<string name="environmental_noise">Ruido ambiental</string>
|
||||||
|
<string name="reconnect_to_last_device">Reconectar al último dispositivo conectado</string>
|
||||||
|
<string name="disconnect">Desconectar</string>
|
||||||
|
<string name="support_me">Apóyame</string>
|
||||||
|
<string name="never_show_again">No volver a mostrar</string>
|
||||||
|
<string name="support_dialog_description">Hace poco perdí mi AirPod izquierdo. Si LibrePods te ha resultado útil, considera apoyarme en GitHub Sponsors para que pueda comprar un reemplazo y seguir trabajando en este proyecto; incluso una pequeña donación es de gran ayuda. ¡Gracias por tu apoyo!</string>
|
||||||
|
<string name="support_librepods">Apoya a LibrePods</string>
|
||||||
|
<string name="listening_mode_off_description">Desactiva la gestión del ruido</string>
|
||||||
|
<string name="listening_mode_transparency_description">Deja entrar los sonidos externos</string>
|
||||||
|
<string name="listening_mode_adaptive_description">Ajuste dinámico del ruido externo</string>
|
||||||
|
<string name="listening_mode_noise_cancellation_description">Bloquea los sonidos externos</string>
|
||||||
|
</resources>
|
||||||
217
android/app/src/main/res/values-fr/strings.xml
Normal file
217
android/app/src/main/res/values-fr/strings.xml
Normal file
@@ -0,0 +1,217 @@
|
|||||||
|
<resources xmlns:tools="http://schemas.android.com/tools" tools:ignore="MissingTranslation">
|
||||||
|
<string name="app_name" translatable="false">LibrePods</string>
|
||||||
|
<string name="app_description">Libérez vos AirPods de l\'écosystème Apple.</string>
|
||||||
|
<string name="app_widget_description">Voyez l\'état de la batterie de vos AirPods directement depuis votre écran d\'accueil !</string>
|
||||||
|
<string name="accessibility">Accessibilité</string>
|
||||||
|
<string name="tone_volume">Volume de la tonalité</string>
|
||||||
|
<string name="tone_volume_description">Ajustez le volume des effets sonores émis sur les AirPods.</string>
|
||||||
|
<string name="audio">Audio</string>
|
||||||
|
<string name="adaptive_audio">Audio Adaptatif</string>
|
||||||
|
<string name="customize_adaptive_audio">Personnaliser l\'Audio Adaptatif</string>
|
||||||
|
<string name="adaptive_audio_description">L\'audio Adaptatif répond dynamiquement à votre environnement et annule ou laisse entrer le bruit extérieur. Vous pouvez personnaliser l\'Audio Adaptatif pour laisser entrer plus ou moins de bruit.</string>
|
||||||
|
<string name="buds">Écouteurs</string>
|
||||||
|
<string name="case_alt">Boîtier</string>
|
||||||
|
<string name="test">Test</string>
|
||||||
|
<string name="name">Nom</string>
|
||||||
|
<string name="noise_control">Mode d\'écoute</string>
|
||||||
|
<string name="off">Désactivé</string>
|
||||||
|
<string name="transparency">Transparence</string>
|
||||||
|
<string name="adaptive">Adaptatif</string>
|
||||||
|
<string name="noise_cancellation">Réduction de bruit</string>
|
||||||
|
<string name="press_and_hold_airpods">Appuyer et maintenir les AirPods</string>
|
||||||
|
<string name="press_and_hold_noise_control_description">Maintenez la tige pour passer entre les modes d\'écoute sélectionnés.</string>
|
||||||
|
<string name="head_gestures">Gestes de la tête</string>
|
||||||
|
<string name="left">Gauche</string>
|
||||||
|
<string name="right">Droite</string>
|
||||||
|
<string name="conversational_awareness">Détection des conversations</string>
|
||||||
|
<string name="conversational_awareness_description">Baisse le volume des médias et réduit le bruit de fond lorsque vous commencez à parler à quelqu\'un.</string>
|
||||||
|
<string name="personalized_volume">Volume personnalisé</string>
|
||||||
|
<string name="personalized_volume_description">Ajuste le volume des médias en réponse à votre environnement.</string>
|
||||||
|
<string name="noise_cancellation_single_airpod">Réduction du bruit avec un écouteur</string>
|
||||||
|
<string name="noise_cancellation_single_airpod_description">Permet d\'activer la réduction de bruit même avec un seul AirPod à l\'oreille.</string>
|
||||||
|
<string name="volume_control">Contrôle du volume</string>
|
||||||
|
<string name="volume_control_description">Ajustez le volume en balayant vers le haut ou vers le bas sur le capteur situé sur la tige des AirPods Pro.</string>
|
||||||
|
<string name="airpods_not_connected">AirPods non connectés</string>
|
||||||
|
<string name="airpods_not_connected_description">Veuillez connecter vos AirPods pour accéder aux réglages.</string>
|
||||||
|
<string name="back">Retour</string>
|
||||||
|
<string name="app_settings">Personnalisations</string>
|
||||||
|
<string name="relative_conversational_awareness_volume">Volume relatif</string>
|
||||||
|
<string name="relative_conversational_awareness_volume_description">Réduit à un pourcentage du volume actuel plutôt qu\'au volume maximum.</string>
|
||||||
|
<string name="conversational_awareness_pause_music">Mettre la musique en pause</string>
|
||||||
|
<string name="conversational_awareness_pause_music_description">Quand vous commencez à parler, la musique sera mise en pause.</string>
|
||||||
|
<string name="appwidget_text">EXEMPLE</string>
|
||||||
|
<string name="add_widget">Ajouter le widget</string>
|
||||||
|
<string name="noise_control_widget_description">Contrôlez le mode de réduction de bruit directement depuis votre écran d\'accueil.</string>
|
||||||
|
<string name="island_connected_text">Connecté</string>
|
||||||
|
<string name="island_connected_remote_text">Connecté à Linux</string>
|
||||||
|
<string name="island_taking_over_text">Connecté</string>
|
||||||
|
<string name="island_moved_to_remote_text">Déplacé vers Linux</string>
|
||||||
|
<string name="island_moved_to_other_device_text">Déplacé vers %1$s</string>
|
||||||
|
<string name="island_moved_to_other_device_reversed_text">Reconnecté depuis la notification</string>
|
||||||
|
<string name="head_tracking">Suivi de la tête</string>
|
||||||
|
<string name="head_gestures_details">Hochez la tête pour répondre aux appels, secouez-la pour refuser.</string>
|
||||||
|
<string name="general_settings_header">Général</string>
|
||||||
|
<string name="qs_click_behavior_title">Action de la tuile dans les réglages rapides</string>
|
||||||
|
<string name="qs_click_behavior_dialog_desc">Afficher le dialogue de contrôle du bruit au toucher.</string>
|
||||||
|
<string name="qs_click_behavior_cycle_desc">Faire défiler les modes au toucher.</string>
|
||||||
|
<string name="developer_options_header">Développeur</string>
|
||||||
|
<string name="more_settings_title">Ouvrir les réglages des AirPods</string>
|
||||||
|
<string name="more_settings_subtitle">Gérer les fonctionnalités et préférences des AirPods</string>
|
||||||
|
<string name="ear_detection">Détection automatique des oreilles</string>
|
||||||
|
<string name="auto_play">Lecture Automatique</string>
|
||||||
|
<string name="auto_pause">Pause Automatique</string>
|
||||||
|
<string name="troubleshooting">Dépannage</string>
|
||||||
|
<string name="troubleshooting_description">Collecter des journaux pour diagnostiquer les problèmes de connexion des AirPods</string>
|
||||||
|
<string name="collect_logs">Collecter les journaux</string>
|
||||||
|
<string name="saved_logs">Journaux enregistrés</string>
|
||||||
|
<string name="no_logs_found">Aucun journal sauvegardé trouvé</string>
|
||||||
|
<string name="takeover_header">Préférences d\'auto-connexion</string>
|
||||||
|
<string name="takeover_airpods_state">Se connecter aux AirPods lorsque leur état est :</string>
|
||||||
|
<string name="takeover_disconnected">Déconnectés</string>
|
||||||
|
<string name="takeover_disconnected_desc">Les AirPods ne sont connectés à aucun appareil</string>
|
||||||
|
<string name="takeover_idle">Inactifs</string>
|
||||||
|
<string name="takeover_idle_desc">Un appareil est connecté à vos AirPods, mais ne lit pas de média et n\'est pas en appel</string>
|
||||||
|
<string name="takeover_music">Lecture de média</string>
|
||||||
|
<string name="takeover_music_desc">Un appareil lit du média via les AirPods</string>
|
||||||
|
<string name="takeover_call">En appel</string>
|
||||||
|
<string name="takeover_call_desc">Un appareil est en appel via les AirPods</string>
|
||||||
|
<string name="takeover_phone_state">Se connecter aux AirPods lorsque votre téléphone :</string>
|
||||||
|
<string name="takeover_ringing_call">Reçoit un appel</string>
|
||||||
|
<string name="takeover_ringing_call_desc">Votre téléphone commence à sonner</string>
|
||||||
|
<string name="takeover_media_start">Démarre la lecture média</string>
|
||||||
|
<string name="takeover_media_start_desc">Votre téléphone commence à lire un média</string>
|
||||||
|
<string name="undo">Annuler</string>
|
||||||
|
<string name="customize_transparency_mode_description">Vous pouvez personnaliser le mode Transparence de vos AirPods Pro pour vous aider à entendre ce qui vous entoure.</string>
|
||||||
|
<string name="loud_sound_reduction_description">La réduction des sons forts peut réduire activement votre exposition aux bruits ambiants forts en mode Transparence et Adaptatif. La réduction des sons forts n\'est pas active en mode Désactivé.</string>
|
||||||
|
<string name="loud_sound_reduction">Réduction des sons forts</string>
|
||||||
|
<string name="call_controls">Contrôle des appels</string>
|
||||||
|
<string name="automatically_connect">Se connecter automatiquement à cet appareil</string>
|
||||||
|
<string name="automatically_connect_description">Quand ce mode est activé, les AirPods essaieront de se connecter automatiquement à cet appareil. Sinon, ils essaieront uniquement de se connecter automatiquement au dernier appareil connecté.</string>
|
||||||
|
<string name="sleep_detection">Mettre en pause l\'écoute au moment de s\'endormir</string>
|
||||||
|
<string name="off_listening_mode">Mode d\'écoute Désactivé</string>
|
||||||
|
<string name="off_listening_mode_description">Quand ce paramètre est activé, les modes d\'écoute incluront une option Désactivé. Les sons forts ne sont pas réduits en mode Désactivé.</string>
|
||||||
|
<string name="microphone">Microphone</string>
|
||||||
|
<string name="microphone_mode">Mode du microphone</string>
|
||||||
|
<string name="microphone_automatic">Automatique</string>
|
||||||
|
<string name="microphone_always_right">Toujours à droite</string>
|
||||||
|
<string name="microphone_always_left">Toujours à gauche</string>
|
||||||
|
<string name="answer_call">Répondre aux appels</string>
|
||||||
|
<string name="mute_unmute">Muet / Activer le son</string>
|
||||||
|
<string name="hang_up">Raccrocher</string>
|
||||||
|
<string name="press_once">Appuyer une fois</string>
|
||||||
|
<string name="press_twice">Appuyer deux fois</string>
|
||||||
|
<string name="hearing_aid">Aide auditive</string>
|
||||||
|
<string name="adjustments">Ajustements</string>
|
||||||
|
<string name="swipe_to_control_amplification">Balayer pour contrôler l\'amplification</string>
|
||||||
|
<string name="swipe_amplification_description">En mode Transparence et sans lecture média, balayez sur les commandes tactiles des AirPods Pro pour augmenter ou diminuer l\'amplification des sons environnants.</string>
|
||||||
|
<string name="transparency_mode">Mode Transparence</string>
|
||||||
|
<string name="customize_transparency_mode">Personnaliser le mode Transparence</string>
|
||||||
|
<string name="press_speed">Vitesse d\'appui</string>
|
||||||
|
<string name="press_speed_description">Ajustez la vitesse requise pour appuyer deux ou trois fois sur vos AirPods.</string>
|
||||||
|
<string name="press_and_hold_duration">Durée d\'appui et de maintien</string>
|
||||||
|
<string name="press_and_hold_duration_description">Ajustez la durée requise pour maintenir la pression sur vos AirPods.</string>
|
||||||
|
<string name="volume_swipe_speed">Vitesse du balayage du volume</string>
|
||||||
|
<string name="volume_swipe_speed_description">Pour éviter les changements de volume involontaires, sélectionnez le délai préféré entre les balayages.</string>
|
||||||
|
<string name="equalizer">Égaliseur</string>
|
||||||
|
<string name="apply_eq_to">Appliquer l\'EQ à</string>
|
||||||
|
<string name="phone">Téléphone</string>
|
||||||
|
<string name="media">Média</string>
|
||||||
|
<string name="band_label">Bande %d</string>
|
||||||
|
<string name="default_option">Par défaut</string>
|
||||||
|
<string name="slower">Plus lent</string>
|
||||||
|
<string name="slowest">Très lent</string>
|
||||||
|
<string name="longer">Plus long</string>
|
||||||
|
<string name="longest">Très long</string>
|
||||||
|
<string name="darker">Plus sombre</string>
|
||||||
|
<string name="brighter">Plus clair</string>
|
||||||
|
<string name="less">Moins</string>
|
||||||
|
<string name="more">Plus</string>
|
||||||
|
<string name="amplification">Amplification</string>
|
||||||
|
<string name="balance">Balance</string>
|
||||||
|
<string name="tone">Tonalité</string>
|
||||||
|
<string name="ambient_noise_reduction">Réduction des bruits ambiants</string>
|
||||||
|
<string name="conversation_boost">Amplificateur de conversation</string>
|
||||||
|
<string name="conversation_boost_description">L\'Amplificateur de conversation concentre les AirPods Pro sur la personne en face de vous, facilitant les conversations en face-à-face.</string>
|
||||||
|
<string name="hearing_aid_description">Les AirPods peuvent utiliser les résultats d\'un test auditif pour améliorer la clarté des voix et des sons autour de vous.\n\nL\'aide auditive est destinée aux personnes ayant une perte auditive légère à modérée.</string>
|
||||||
|
<string name="media_assist">Aide multimédia</string>
|
||||||
|
<string name="media_assist_description">Les AirPods Pro peuvent utiliser les résultats d\'un test auditif pour améliorer la clarté de la musique, des vidéos et des appels.</string>
|
||||||
|
<string name="adjust_media">Ajuster la musique et les vidéos</string>
|
||||||
|
<string name="adjust_calls">Ajuster les appels</string>
|
||||||
|
<string name="widget">Widget</string>
|
||||||
|
<string name="show_phone_battery_in_widget">Afficher la batterie du téléphone dans le widget</string>
|
||||||
|
<string name="show_phone_battery_in_widget_description">Affiche le niveau de batterie du téléphone dans le widget avec celle des AirPods</string>
|
||||||
|
<string name="conversational_awareness_volume">Volume de détection des conversations</string>
|
||||||
|
<string name="quick_settings_tile">Tuile des réglages rapides</string>
|
||||||
|
<string name="open_dialog_for_controlling">Ouvrir le dialogue de contrôle</string>
|
||||||
|
<string name="open_dialog_for_controlling_description">Si désactivé, appuyer sur la tuile fera défiler les modes. Si activé, un dialogue apparaîtra pour contrôler le mode d\'écoute et la détection de conversation.</string>
|
||||||
|
<string name="disconnect_when_not_wearing">Déconnecter les AirPods quand ils ne sont pas portés</string>
|
||||||
|
<string name="disconnect_when_not_wearing_description">Vous pourrez toujours les contrôler depuis l\'app — cela déconnecte juste l\'audio.</string>
|
||||||
|
<string name="advanced_options">Options avancées</string>
|
||||||
|
<string name="set_identity_resolving_key">Définir la clé d\'identité et de résolution (IRK)</string>
|
||||||
|
<string name="set_identity_resolving_key_description">Définir manuellement la clé IRK utilisée pour la résolution des adresses BLE aléatoires</string>
|
||||||
|
<string name="set_encryption_key">Définir la clé de chiffrement</string>
|
||||||
|
<string name="set_encryption_key_description">Définir manuellement la clé ENC_KEY utilisée pour déchiffrer les publicités BLE</string>
|
||||||
|
<string name="use_alternate_head_tracking_packets">Utiliser des paquets alternatifs pour le suivi de la tête</string>
|
||||||
|
<string name="use_alternate_head_tracking_packets_description">Activez ceci si le suivi de tête ne fonctionne pas. Cela envoie un autre type de données aux AirPods pour demander/arrêter le suivi de la tête.</string>
|
||||||
|
<string name="act_as_an_apple_device">Se comporter comme un appareil Apple</string>
|
||||||
|
<string name="act_as_an_apple_device_description">Active la connectivité multi-appareils et les fonctionnalités d\'accessibilité comme la personnalisation du mode transparence (amplification, tonalité, réduction de bruit ambiant, amplificateur de conversations, EQ)</string>
|
||||||
|
<string name="act_as_an_apple_device_warning">Peut être instable !! Un maximum de deux appareils peut être connecté à vos AirPods. Si vous utilisez un appareil Apple comme un iPad ou un Mac, connectez-le d\'abord, puis connectez votre Android.</string>
|
||||||
|
<string name="reset_hook_offset">Réinitialiser l\'offset du hook</string>
|
||||||
|
<string name="reset_hook_offset_description">Cela effacera l\'offset actuel et nécessitera de refaire la configuration. Voulez-vous vraiment continuer ?</string>
|
||||||
|
<string name="reset">Réinitialiser</string>
|
||||||
|
<string name="hook_offset_reset_success">Hook offset réinitialisé. Redirection vers la configuration…</string>
|
||||||
|
<string name="hook_offset_reset_failure">Impossible de réinitialiser l\'hook offset</string>
|
||||||
|
<string name="irk_set_success">Clé IRK définie avec succès</string>
|
||||||
|
<string name="encryption_key_set_success">Clé de chiffrement définie avec succès</string>
|
||||||
|
<string name="irk_hex_value">Valeur hex IRK</string>
|
||||||
|
<string name="enc_key_hex_value">Valeur hex ENC_KEY</string>
|
||||||
|
<string name="enter_irk_hex">Entrez l\'IRK de 16 octets en string hexadécimal (32 caractères) :</string>
|
||||||
|
<string name="enter_enc_key_hex">Entrez l\'ENC_KEY de 16 octets en string hexadécimal (32 caractères) :</string>
|
||||||
|
<string name="must_be_32_hex_chars">Doit contenir exactement 32 caractères hexadécimaux</string>
|
||||||
|
<string name="error_converting_hex">Erreur lors de la conversion hexadécimale :</string>
|
||||||
|
<string name="found_offset_restart_bluetooth">Offset trouvé. Veuillez redémarrer le processus Bluetooth.</string>
|
||||||
|
<string name="digital_assistant">Assistant numérique</string>
|
||||||
|
<string name="on">Activé</string>
|
||||||
|
<string name="camera_remote">Télécommande de l\'appareil photo</string>
|
||||||
|
<string name="camera_control">Contrôle de la caméra</string>
|
||||||
|
<string name="camera_control_description">Prenez une photo, lancez/arrêtez un enregistrement, etc., avec un appui simple ou un appui long. Si vous utilisez un appui simple, les gestes de contrôle multimédia seront indisponibles ; si vous utilisez un appui long, les gestes de mode d\'écoute et d\'assistant numérique seront indisponibles.</string>
|
||||||
|
<string name="camera_control_app_description">Définir un paquet d\'application de caméra personnalisé</string>
|
||||||
|
<string name="set_custom_camera_package">Définir un appid de caméra personnalisé</string>
|
||||||
|
<string name="enter_custom_camera_package">Entrez l\'identifiant de l\'application caméra :</string>
|
||||||
|
<string name="custom_camera_package">Appid caméra personnalisé</string>
|
||||||
|
<string name="custom_camera_package_set_success">Appid caméra personnalisé défini avec succès</string>
|
||||||
|
<string name="app_listener_service_label">Service d\'écoute de la caméra</string>
|
||||||
|
<string name="app_listener_service_description">Service d\'écoute pour que LibrePods détecte quand la caméra est active afin d\'activer le contrôle caméra via les AirPods.</string>
|
||||||
|
<string name="open_source_licenses">Licences open source</string>
|
||||||
|
<string name="hearing_test">Mettre à jour le test d\'audition</string>
|
||||||
|
<string name="update_hearing_test">Mettre à jour les résultats du test d\'audition</string>
|
||||||
|
<string name="att_manager_is_null_try_reconnecting">ATT Manager est null, essayez de reconnecter.</string>
|
||||||
|
<string name="permissions_required">Les permissions suivantes sont requises pour utiliser l\'application. Veuillez les accorder pour continuer.</string>
|
||||||
|
<string name="shake_your_head_or_nod">Secouez la tête ou hochez-la !</string>
|
||||||
|
<string name="root_access_required">Accès root requis</string>
|
||||||
|
<string name="this_app_needs_root_access_to_hook_onto_the_bluetooth_library">Cette application nécessite l\'accès root pour s\'injecter dans la bibliothèque Bluetooth.</string>
|
||||||
|
<string name="root_access_denied">Accès root refusé. Veuillez accorder les permissions root.</string>
|
||||||
|
<string name="troubleshooting_steps">Étapes de dépannage</string>
|
||||||
|
<string name="hearing_test_value_instruction">Veuillez entrer les valeurs de perte en dBHL</string>
|
||||||
|
<string name="about">À propos</string>
|
||||||
|
<string name="model_name">Nom du modèle</string>
|
||||||
|
<string name="model_number">Numéro du modèle</string>
|
||||||
|
<string name="serial_number">Numéro de série</string>
|
||||||
|
<string name="version">Version</string>
|
||||||
|
<string name="hearing_health">Santé auditive</string>
|
||||||
|
<string name="hearing_protection">Protection auditive</string>
|
||||||
|
<string name="workspace_use">Usage en environnement de travail</string>
|
||||||
|
<string name="ppe">Protection EN 352</string>
|
||||||
|
<string name="workspace_use_description">La protection EN 352 limite le niveau sonore maximal à 82 dBA et répond aux exigences applicables de la norme EN 352 pour la protection auditive personnelle.</string>
|
||||||
|
<string name="environmental_noise">Bruit environnemental</string>
|
||||||
|
<string name="reconnect_to_last_device">Reconnecter au dernier appareil</string>
|
||||||
|
<string name="disconnect">Déconnecter</string>
|
||||||
|
<string name="support_me">Soutenez-moi</string>
|
||||||
|
<string name="never_show_again">Ne plus afficher</string>
|
||||||
|
<string name="support_dialog_description">J\'ai récemment perdu mon AirPod gauche. Si LibrePods vous est utile, pensez à me soutenir sur GitHub Sponsors pour m\'aider à en racheter un et continuer ce projet — même un petit montant aide beaucoup. Merci pour votre soutien !</string>
|
||||||
|
<string name="support_librepods">Soutenir LibrePods</string>
|
||||||
|
<string name="listening_mode_off_description">Désactiver la gestion du bruit</string>
|
||||||
|
<string name="listening_mode_transparency_description">Laisser entrer les sons extérieurs</string>
|
||||||
|
<string name="listening_mode_adaptive_description">Ajuster dynamiquement les sons extérieurs</string>
|
||||||
|
<string name="listening_mode_noise_cancellation_description">Bloquer les sons extérieurs</string>
|
||||||
|
</resources>
|
||||||
217
android/app/src/main/res/values-pt/strings.xml
Normal file
217
android/app/src/main/res/values-pt/strings.xml
Normal file
@@ -0,0 +1,217 @@
|
|||||||
|
<resources xmlns:tools="http://schemas.android.com/tools" tools:ignore="MissingTranslation">
|
||||||
|
<string name="app_name" translatable="false">LibrePods</string>
|
||||||
|
<string name="app_description">Libere seus AirPods do ecossistema da Apple.</string>
|
||||||
|
<string name="app_widget_description">Veja o status da bateria dos seus AirPods diretamente na tela inicial!</string>
|
||||||
|
<string name="accessibility">Acessibilidade</string>
|
||||||
|
<string name="tone_volume">Volume do Tom</string>
|
||||||
|
<string name="tone_volume_description">Ajuste o volume do tom dos efeitos sonoros reproduzidos pelos AirPods.</string>
|
||||||
|
<string name="audio">Áudio</string>
|
||||||
|
<string name="adaptive_audio">Áudio Adaptativo</string>
|
||||||
|
<string name="customize_adaptive_audio">Personalizar Áudio Adaptativo</string>
|
||||||
|
<string name="adaptive_audio_description">O áudio adaptativo responde dinamicamente ao seu ambiente e cancela ou permite ruídos externos. Você pode personalizar o Áudio Adaptativo para permitir mais ou menos ruído.</string>
|
||||||
|
<string name="buds">Fones</string>
|
||||||
|
<string name="case_alt">Estojo</string>
|
||||||
|
<string name="test">Teste</string>
|
||||||
|
<string name="name">Nome</string>
|
||||||
|
<string name="noise_control">Modo de Escuta</string>
|
||||||
|
<string name="off">Desligado</string>
|
||||||
|
<string name="transparency">Transparência</string>
|
||||||
|
<string name="adaptive">Adaptativo</string>
|
||||||
|
<string name="noise_cancellation">Cancelamento de Ruído</string>
|
||||||
|
<string name="press_and_hold_airpods">Pressionar e Segurar AirPods</string>
|
||||||
|
<string name="press_and_hold_noise_control_description">Pressione e segure a haste para alternar entre os modos de escuta selecionados.</string>
|
||||||
|
<string name="head_gestures">Gestos com a Cabeça</string>
|
||||||
|
<string name="left">Esquerdo</string>
|
||||||
|
<string name="right">Direito</string>
|
||||||
|
<string name="conversational_awareness">Consciência Conversacional</string>
|
||||||
|
<string name="conversational_awareness_description">Reduz o volume da mídia e diminui o ruído de fundo quando você começa a falar com outras pessoas.</string>
|
||||||
|
<string name="personalized_volume">Volume Personalizado</string>
|
||||||
|
<string name="personalized_volume_description">Ajusta o volume da mídia em resposta ao seu ambiente.</string>
|
||||||
|
<string name="noise_cancellation_single_airpod">Cancelamento de Ruído com um AirPod</string>
|
||||||
|
<string name="noise_cancellation_single_airpod_description">Permite que os AirPods sejam colocados em modo de cancelamento de ruído quando apenas um AirPod está no seu ouvido.</string>
|
||||||
|
<string name="volume_control">Controle de Volume</string>
|
||||||
|
<string name="volume_control_description">Ajuste o volume deslizando para cima ou para baixo no sensor localizado na haste dos AirPods Pro.</string>
|
||||||
|
<string name="airpods_not_connected">AirPods não conectados</string>
|
||||||
|
<string name="airpods_not_connected_description">Por favor, conecte seus AirPods para acessar as configurações.</string>
|
||||||
|
<string name="back">Voltar</string>
|
||||||
|
<string name="app_settings">Personalizações</string>
|
||||||
|
<string name="relative_conversational_awareness_volume">Volume relativo</string>
|
||||||
|
<string name="relative_conversational_awareness_volume_description">Reduz para uma porcentagem do volume atual em vez do volume máximo.</string>
|
||||||
|
<string name="conversational_awareness_pause_music">Pausar Música</string>
|
||||||
|
<string name="conversational_awareness_pause_music_description">Quando você começar a falar, a música será pausada.</string>
|
||||||
|
<string name="appwidget_text">EXEMPLO</string>
|
||||||
|
<string name="add_widget">Adicionar widget</string>
|
||||||
|
<string name="noise_control_widget_description">Controle o Modo de Controle de Ruído diretamente da sua Tela Inicial.</string>
|
||||||
|
<string name="island_connected_text">Conectado</string>
|
||||||
|
<string name="island_connected_remote_text">Conectado ao Linux</string>
|
||||||
|
<string name="island_taking_over_text">Conectado</string>
|
||||||
|
<string name="island_moved_to_remote_text">Movido para Linux</string>
|
||||||
|
<string name="island_moved_to_other_device_text">Movido para %1$s</string>
|
||||||
|
<string name="island_moved_to_other_device_reversed_text">Reconectar pela notificação</string>
|
||||||
|
<string name="head_tracking">Rastreamento de Cabeça</string>
|
||||||
|
<string name="head_gestures_details">Acene para atender chamadas e balance a cabeça para recusar.</string>
|
||||||
|
<string name="general_settings_header">Geral</string>
|
||||||
|
<string name="qs_click_behavior_title">Ação do Bloco de Configurações Rápidas</string>
|
||||||
|
<string name="qs_click_behavior_dialog_desc">Mostrar diálogo de controle de ruído ao tocar.</string>
|
||||||
|
<string name="qs_click_behavior_cycle_desc">Alternar entre modos ao tocar.</string>
|
||||||
|
<string name="developer_options_header">Desenvolvedor</string>
|
||||||
|
<string name="more_settings_title">Abrir Configurações dos AirPods</string>
|
||||||
|
<string name="more_settings_subtitle">Gerencie recursos e preferências dos AirPods</string>
|
||||||
|
<string name="ear_detection">Detecção Automática de Ouvido</string>
|
||||||
|
<string name="auto_play">Reprodução Automática</string>
|
||||||
|
<string name="auto_pause">Pausa Automática</string>
|
||||||
|
<string name="troubleshooting">Solução de Problemas</string>
|
||||||
|
<string name="troubleshooting_description">Coletar logs para diagnosticar problemas com a conexão dos AirPods</string>
|
||||||
|
<string name="collect_logs">Coletar Logs</string>
|
||||||
|
<string name="saved_logs">Logs Salvos</string>
|
||||||
|
<string name="no_logs_found">Nenhum log salvo encontrado</string>
|
||||||
|
<string name="takeover_header">Preferências de Auto-conexão</string>
|
||||||
|
<string name="takeover_airpods_state">Conectar aos seus AirPods quando o status for:</string>
|
||||||
|
<string name="takeover_disconnected">Desconectado</string>
|
||||||
|
<string name="takeover_disconnected_desc">Os AirPods não estão conectados a um dispositivo</string>
|
||||||
|
<string name="takeover_idle">Inativo</string>
|
||||||
|
<string name="takeover_idle_desc">Um dispositivo está conectado aos seus AirPods, mas não está reproduzindo mídia ou em uma chamada</string>
|
||||||
|
<string name="takeover_music">Reproduzindo mídia</string>
|
||||||
|
<string name="takeover_music_desc">Um dispositivo está reproduzindo mídia nos seus AirPods</string>
|
||||||
|
<string name="takeover_call">Em chamada</string>
|
||||||
|
<string name="takeover_call_desc">Um dispositivo está em uma chamada com seus AirPods</string>
|
||||||
|
<string name="takeover_phone_state">Conectar aos AirPods quando seu telefone estiver:</string>
|
||||||
|
<string name="takeover_ringing_call">Recebendo uma chamada</string>
|
||||||
|
<string name="takeover_ringing_call_desc">Seu telefone começa a tocar</string>
|
||||||
|
<string name="takeover_media_start">Iniciando reprodução de mídia</string>
|
||||||
|
<string name="takeover_media_start_desc">Seu telefone começa a reproduzir mídia</string>
|
||||||
|
<string name="undo">Desfazer</string>
|
||||||
|
<string name="customize_transparency_mode_description">Você pode personalizar o modo de Transparência para seus AirPods Pro para ajudá-lo a ouvir o que está ao seu redor.</string>
|
||||||
|
<string name="loud_sound_reduction_description">A Redução de Som Alto pode reduzir ativamente sua exposição a ruídos ambientais altos quando estiver nos modos Transparência e Adaptativo. A Redução de Som Alto não está ativa no modo Desligado.</string>
|
||||||
|
<string name="loud_sound_reduction">Redução de Som Alto</string>
|
||||||
|
<string name="call_controls">Controles de Chamada</string>
|
||||||
|
<string name="automatically_connect">Conectar a este dispositivo automaticamente</string>
|
||||||
|
<string name="automatically_connect_description">Quando habilitado, os AirPods tentarão conectar a este dispositivo automaticamente. Caso contrário, eles tentarão conectar automaticamente apenas quando conectados pela última vez.</string>
|
||||||
|
<string name="sleep_detection">Pausar mídia ao adormecer</string>
|
||||||
|
<string name="off_listening_mode">Modo de Escuta Desligado</string>
|
||||||
|
<string name="off_listening_mode_description">Quando isso estiver ativado, os modos de escuta dos AirPods incluirão uma opção Desligado. Os níveis de som alto não são reduzidos quando o modo de escuta está definido como Desligado.</string>
|
||||||
|
<string name="microphone">Microfone</string>
|
||||||
|
<string name="microphone_mode">Modo do Microfone</string>
|
||||||
|
<string name="microphone_automatic">Automático</string>
|
||||||
|
<string name="microphone_always_right">Sempre Direito</string>
|
||||||
|
<string name="microphone_always_left">Sempre Esquerdo</string>
|
||||||
|
<string name="answer_call">Atender chamada</string>
|
||||||
|
<string name="mute_unmute">Silenciar/Ativar Som</string>
|
||||||
|
<string name="hang_up">Desligar</string>
|
||||||
|
<string name="press_once">Pressionar Uma Vez</string>
|
||||||
|
<string name="press_twice">Pressionar Duas Vezes</string>
|
||||||
|
<string name="hearing_aid">Auxiliar de Audição</string>
|
||||||
|
<string name="adjustments">Ajustes</string>
|
||||||
|
<string name="swipe_to_control_amplification">Deslize para controlar a amplificação</string>
|
||||||
|
<string name="swipe_amplification_description">Quando estiver no modo Transparência e nenhuma mídia estiver sendo reproduzida, deslize para cima e para baixo nos controles de toque dos seus AirPods Pro para aumentar ou diminuir a amplificação dos sons ambientais.</string>
|
||||||
|
<string name="transparency_mode">Modo Transparência</string>
|
||||||
|
<string name="customize_transparency_mode">Personalizar Modo Transparência</string>
|
||||||
|
<string name="press_speed">Velocidade de Pressionamento</string>
|
||||||
|
<string name="press_speed_description">Ajuste a velocidade necessária para pressionar duas ou três vezes nos seus AirPods.</string>
|
||||||
|
<string name="press_and_hold_duration">Duração de Pressionar e Segurar</string>
|
||||||
|
<string name="press_and_hold_duration_description">Ajuste a duração necessária para pressionar e segurar nos seus AirPods.</string>
|
||||||
|
<string name="volume_swipe_speed">Velocidade de Deslize de Volume</string>
|
||||||
|
<string name="volume_swipe_speed_description">Para evitar ajustes de volume não intencionais, selecione o tempo de espera preferido entre deslizes.</string>
|
||||||
|
<string name="equalizer">Equalizador</string>
|
||||||
|
<string name="apply_eq_to">Aplicar EQ a</string>
|
||||||
|
<string name="phone">Telefone</string>
|
||||||
|
<string name="media">Mídia</string>
|
||||||
|
<string name="band_label">Banda %d</string>
|
||||||
|
<string name="default_option">Padrão</string>
|
||||||
|
<string name="slower">Mais Lento</string>
|
||||||
|
<string name="slowest">Mais Lento</string>
|
||||||
|
<string name="longer">Mais Longo</string>
|
||||||
|
<string name="longest">Mais Longo</string>
|
||||||
|
<string name="darker">Mais Escuro</string>
|
||||||
|
<string name="brighter">Mais Claro</string>
|
||||||
|
<string name="less">Menos</string>
|
||||||
|
<string name="more">Mais</string>
|
||||||
|
<string name="amplification">Amplificação</string>
|
||||||
|
<string name="balance">Balanço</string>
|
||||||
|
<string name="tone">Tom</string>
|
||||||
|
<string name="ambient_noise_reduction">Redução de Ruído Ambiental</string>
|
||||||
|
<string name="conversation_boost">Amplificação de Conversa</string>
|
||||||
|
<string name="conversation_boost_description">A Amplificação de Conversa foca seus AirPods Pro na pessoa falando na sua frente, facilitando ouvir em uma conversa face a face.</string>
|
||||||
|
<string name="hearing_aid_description">Os AirPods podem usar os resultados de um teste auditivo para fazer ajustes que melhoram a clareza de vozes e sons ao seu redor.\n\nO Auxiliar de Audição é destinado apenas para pessoas com perda auditiva leve a moderada percebida.</string>
|
||||||
|
<string name="media_assist">Assistente de Mídia</string>
|
||||||
|
<string name="media_assist_description">Os AirPods Pro podem usar os resultados de um teste auditivo para fazer ajustes que melhoram a clareza de música, vídeo e chamadas.</string>
|
||||||
|
<string name="adjust_media">Ajustar Música e Vídeo</string>
|
||||||
|
<string name="adjust_calls">Ajustar Chamadas</string>
|
||||||
|
<string name="widget">Widget</string>
|
||||||
|
<string name="show_phone_battery_in_widget">Mostrar bateria do telefone no widget</string>
|
||||||
|
<string name="show_phone_battery_in_widget_description">Exiba o nível de bateria do seu telefone no widget junto com a bateria dos AirPods</string>
|
||||||
|
<string name="conversational_awareness_volume">Volume de Consciência Conversacional</string>
|
||||||
|
<string name="quick_settings_tile">Bloco de Configurações Rápidas</string>
|
||||||
|
<string name="open_dialog_for_controlling">Abrir diálogo para controlar</string>
|
||||||
|
<string name="open_dialog_for_controlling_description">Se desabilitado, clicar no bloco de configurações rápidas alternará entre modos. Se habilitado, mostrará um diálogo para controlar o modo de controle de ruído e a consciência conversacional</string>
|
||||||
|
<string name="disconnect_when_not_wearing">Desconectar AirPods quando não estiver usando</string>
|
||||||
|
<string name="disconnect_when_not_wearing_description">Você ainda poderá controlá-los com o aplicativo - isso apenas desconecta o áudio.</string>
|
||||||
|
<string name="advanced_options">Opções Avançadas</string>
|
||||||
|
<string name="set_identity_resolving_key">Definir Chave de Resolução de Identidade (IRK)</string>
|
||||||
|
<string name="set_identity_resolving_key_description">Defina manualmente o valor IRK usado para resolver endereços aleatórios BLE</string>
|
||||||
|
<string name="set_encryption_key">Definir Chave de Criptografia</string>
|
||||||
|
<string name="set_encryption_key_description">Defina manualmente o valor ENC_KEY usado para descriptografar anúncios BLE</string>
|
||||||
|
<string name="use_alternate_head_tracking_packets">Usar pacotes alternativos de rastreamento de cabeça</string>
|
||||||
|
<string name="use_alternate_head_tracking_packets_description">Habilite isso se o rastreamento de cabeça não funcionar para você. Isso envia dados diferentes para os AirPods para solicitar/parar dados de rastreamento de cabeça.</string>
|
||||||
|
<string name="act_as_an_apple_device">Agir como um dispositivo Apple</string>
|
||||||
|
<string name="act_as_an_apple_device_description">Habilita conectividade multi-dispositivo e recursos de Acessibilidade como personalização do modo de transparência (amplificação, tom, redução de ruído ambiental, amplificação de conversa e equalizador)</string>
|
||||||
|
<string name="act_as_an_apple_device_warning">Pode ser instável!! Um máximo de dois dispositivos pode estar conectado aos seus AirPods. Se você estiver usando com um dispositivo Apple como iPad ou Mac, então conecte esse dispositivo primeiro e depois seu Android.</string>
|
||||||
|
<string name="reset_hook_offset">Redefinir Offset do Hook</string>
|
||||||
|
<string name="reset_hook_offset_description">Isso limpará o offset do hook atual e exigirá que você passe pelo processo de configuração novamente. Tem certeza de que deseja continuar?</string>
|
||||||
|
<string name="reset">Redefinir</string>
|
||||||
|
<string name="hook_offset_reset_success">O offset do hook foi redefinido. Redirecionando para a configuração...</string>
|
||||||
|
<string name="hook_offset_reset_failure">Falha ao redefinir o offset do hook</string>
|
||||||
|
<string name="irk_set_success">IRK foi definido com sucesso</string>
|
||||||
|
<string name="encryption_key_set_success">A chave de criptografia foi definida com sucesso</string>
|
||||||
|
<string name="irk_hex_value">Valor Hexadecimal IRK</string>
|
||||||
|
<string name="enc_key_hex_value">Valor Hexadecimal ENC_KEY</string>
|
||||||
|
<string name="enter_irk_hex">Digite o IRK de 16 bytes como string hexadecimal (32 caracteres):</string>
|
||||||
|
<string name="enter_enc_key_hex">Digite o ENC_KEY de 16 bytes como string hexadecimal (32 caracteres):</string>
|
||||||
|
<string name="must_be_32_hex_chars">Deve ter exatamente 32 caracteres hexadecimais</string>
|
||||||
|
<string name="error_converting_hex">Erro ao converter hexadecimal:</string>
|
||||||
|
<string name="found_offset_restart_bluetooth">Offset encontrado, por favor reinicie o processo Bluetooth</string>
|
||||||
|
<string name="digital_assistant">Assistente Digital</string>
|
||||||
|
<string name="on">Ligado</string>
|
||||||
|
<string name="camera_remote">Controle Remoto da Câmera</string>
|
||||||
|
<string name="camera_control">Controle da Câmera</string>
|
||||||
|
<string name="camera_control_description">Capture uma foto, inicie ou pare a gravação e mais usando Pressionar Uma Vez ou Pressionar e Segurar. Ao usar AirPods para ações da câmera, se você selecionar Pressionar Uma Vez, os gestos de controle de mídia estarão indisponíveis, e se você selecionar Pressionar e Segurar, os gestos de modo de escuta e Assistente Digital estarão indisponíveis.</string>
|
||||||
|
<string name="camera_control_app_description">Defina um pacote de aplicativo personalizado para detecção de câmera</string>
|
||||||
|
<string name="set_custom_camera_package">Definir ID do Aplicativo de Câmera Personalizado</string>
|
||||||
|
<string name="enter_custom_camera_package">Digite o ID do aplicativo da câmera:</string>
|
||||||
|
<string name="custom_camera_package">ID do Aplicativo de Câmera Personalizado</string>
|
||||||
|
<string name="custom_camera_package_set_success">ID do aplicativo de câmera personalizado definido com sucesso</string>
|
||||||
|
<string name="app_listener_service_label">Ouvinte de câmera</string>
|
||||||
|
<string name="app_listener_service_description">Serviço de ouvinte do LibrePods para detectar quando a câmera está ativa para ativar o controle da câmera nos AirPods.</string>
|
||||||
|
<string name="open_source_licenses">Licenças de Código Aberto</string>
|
||||||
|
<string name="hearing_test">Atualizar Teste Auditivo</string>
|
||||||
|
<string name="update_hearing_test">Atualizar Resultado do Teste Auditivo</string>
|
||||||
|
<string name="att_manager_is_null_try_reconnecting">O gerenciador ATT está nulo, tente reconectar.</string>
|
||||||
|
<string name="permissions_required">As seguintes permissões são necessárias para usar o aplicativo. Por favor, conceda-as para continuar.</string>
|
||||||
|
<string name="shake_your_head_or_nod">Balance a cabeça ou acene!</string>
|
||||||
|
<string name="root_access_required">Acesso Root Necessário</string>
|
||||||
|
<string name="this_app_needs_root_access_to_hook_onto_the_bluetooth_library">Este aplicativo precisa de acesso root para conectar-se à biblioteca Bluetooth</string>
|
||||||
|
<string name="root_access_denied">O acesso root foi negado. Por favor, conceda permissões root.</string>
|
||||||
|
<string name="troubleshooting_steps">Etapas de Solução de Problemas</string>
|
||||||
|
<string name="hearing_test_value_instruction">Por favor, digite os valores de perda em dbHL</string>
|
||||||
|
<string name="about">Sobre</string>
|
||||||
|
<string name="model_name">Nome do Modelo</string>
|
||||||
|
<string name="model_number">Número do Modelo</string>
|
||||||
|
<string name="serial_number">Número de Série</string>
|
||||||
|
<string name="version">Versão</string>
|
||||||
|
<string name="hearing_health">Saúde Auditiva</string>
|
||||||
|
<string name="hearing_protection">Proteção Auditiva</string>
|
||||||
|
<string name="workspace_use">Uso no Local de Trabalho</string>
|
||||||
|
<string name="ppe">Proteção EN 352</string>
|
||||||
|
<string name="workspace_use_description">A Proteção EN 352 limita o nível máximo de mídia a 82 dBA e atende aos requisitos aplicáveis do padrão EN 352 para proteção auditiva pessoal.</string>
|
||||||
|
<string name="environmental_noise">Ruído Ambiental</string>
|
||||||
|
<string name="reconnect_to_last_device">Reconectar ao último dispositivo conectado</string>
|
||||||
|
<string name="disconnect">Desconectar</string>
|
||||||
|
<string name="support_me">Me Apoiar</string>
|
||||||
|
<string name="never_show_again">Nunca mostrar novamente</string>
|
||||||
|
<string name="support_dialog_description">Recentemente perdi meu AirPod esquerdo. Se você achou o LibrePods útil, considere me apoiar no GitHub Sponsors para que eu possa comprar uma substituição e continuar trabalhando neste projeto - mesmo uma pequena quantia faz muita diferença. Obrigado pelo seu apoio!</string>
|
||||||
|
<string name="support_librepods">Apoiar LibrePods</string>
|
||||||
|
<string name="listening_mode_off_description">Desativa o gerenciamento de ruído</string>
|
||||||
|
<string name="listening_mode_transparency_description">Permite sons externos</string>
|
||||||
|
<string name="listening_mode_adaptive_description">Ajusta dinamicamente o ruído externo</string>
|
||||||
|
<string name="listening_mode_noise_cancellation_description">Bloqueia sons externos</string>
|
||||||
|
</resources>
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user