Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

psyqo Multi-tap driver and example update #1815

Open
wants to merge 13 commits into
base: main
Choose a base branch
from

Conversation

johnbaumann
Copy link
Collaborator

@johnbaumann johnbaumann commented Dec 13, 2024

Add getters for adc+pad type, update multitap example

@johnbaumann
Copy link
Collaborator Author

I really didn't know what sort of access methods would make sense for the analog data fields here. I can scrap this and try again if this one's too painful to look at, otherwise, open to suggestions.

@nicolasnoble
Copy link
Member

Yeah no I was just trying to avoid doing too much during the holidays :P

@johnbaumann
Copy link
Collaborator Author

@coderabbitai full review

Copy link
Contributor

coderabbitai bot commented Jan 5, 2025

✅ Actions performed

Full review triggered.

Copy link
Contributor

coderabbitai bot commented Jan 5, 2025

Walkthrough

The changes enhance the AdvancedPad class in the PsyQo library by updating the copyright year and modifying the class structure and functionality. This includes changing the Pad enum to a scoped enum class, moving the Command enum to the private section, and adding several methods for handling analog input states. The multitap.cpp example file is also updated to utilize these new methods, improving the display of connected pad information, including types and statuses.

Changes

File Changes
src/mips/psyqo/advancedpad.hh - Updated copyright year
- Changed Pad enum to scoped enum class
- Moved Command enum to private section
- Added methods: getAdc(), getHalfword(), getPadType()
- Updated isPadConnected() and isButtonPressed() methods for type safety
- Modified m_padData structure to a union
src/mips/psyqo/examples/multitap/multitap.cpp - Added printPadType() method
- Removed printPadConnectionStatus(), replaced with printPadList()
- Updated printPadStatus() to remove name parameter
- Refactored print() method for flexible formatting
- Adjusted printed output layout
src/mips/psyqo/src/advancedpad.cpp - Updated pad identifier references to namespaced format
- Changed handling of m_connected and m_buttons arrays for type safety
- Updated copyright year

Sequence Diagram

sequenceDiagram
    participant Pad as Pad Device
    participant AdvancedPad as AdvancedPad Class
    participant MultitapScene as Multitap Scene

    Pad ->> AdvancedPad: Provide Pad Data
    AdvancedPad ->> AdvancedPad: Process Pad Type
    AdvancedPad ->> AdvancedPad: Extract Analog Inputs
    AdvancedPad ->> MultitapScene: Return Pad Information
    MultitapScene ->> MultitapScene: Display Pad Details
Loading

Poem

🎮 Analog Whiskers Twitch

Pads dance with digital glee,
Inputs flowing, secrets free,
Bytes and signals, rabbit's delight,
Multitap magic shining bright!

🐰✨


🪧 Tips

Chat

There are 3 ways to chat with CodeRabbit:

  • Review comments: Directly reply to a review comment made by CodeRabbit. Example:
    • I pushed a fix in commit <commit_id>, please review it.
    • Generate unit testing code for this file.
    • Open a follow-up GitHub issue for this discussion.
  • Files and specific lines of code (under the "Files changed" tab): Tag @coderabbitai in a new review comment at the desired location with your query. Examples:
    • @coderabbitai generate unit testing code for this file.
    • @coderabbitai modularize this function.
  • PR comments: Tag @coderabbitai in a new PR comment to ask questions about the PR branch. For the best results, please provide a very specific query, as very limited context is provided in this mode. Examples:
    • @coderabbitai gather interesting stats about this repository and render them as a table. Additionally, render a pie chart showing the language distribution in the codebase.
    • @coderabbitai read src/utils.ts and generate unit testing code.
    • @coderabbitai read the files in the src/scheduler package and generate a class diagram using mermaid and a README in the markdown format.
    • @coderabbitai help me debug CodeRabbit configuration file.

Note: Be mindful of the bot's finite context window. It's strongly recommended to break down tasks such as reading entire modules into smaller chunks. For a focused discussion, use review comments to chat about specific files and their changes, instead of using the PR comments.

CodeRabbit Commands (Invoked using PR comments)

  • @coderabbitai pause to pause the reviews on a PR.
  • @coderabbitai resume to resume the paused reviews.
  • @coderabbitai review to trigger an incremental review. This is useful when automatic reviews are disabled for the repository.
  • @coderabbitai full review to do a full review from scratch and review all the files again.
  • @coderabbitai summary to regenerate the summary of the PR.
  • @coderabbitai generate docstrings to generate docstrings for this PR. (Beta)
  • @coderabbitai resolve resolve all the CodeRabbit review comments.
  • @coderabbitai configuration to show the current CodeRabbit configuration for the repository.
  • @coderabbitai help to get help.

Other keywords and placeholders

  • Add @coderabbitai ignore anywhere in the PR description to prevent this PR from being reviewed.
  • Add @coderabbitai summary to generate the high-level summary at a specific location in the PR description.
  • Add @coderabbitai anywhere in the PR title to generate the title automatically.

CodeRabbit Configuration File (.coderabbit.yaml)

  • You can programmatically configure CodeRabbit by adding a .coderabbit.yaml file to the root of your repository.
  • Please see the configuration documentation for more information.
  • If your editor has YAML language server enabled, you can add the path at the top of this file to enable auto-completion and validation: # yaml-language-server: $schema=https://coderabbit.ai/integrations/schema.v2.json

Documentation and Community

  • Visit our Documentation for detailed information on how to use CodeRabbit.
  • Join our Discord Community to get help, request features, and share feedback.
  • Follow us on X/Twitter for updates and announcements.

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 1

🧹 Nitpick comments (2)
src/mips/psyqo/examples/multitap/multitap.cpp (2)

112-112: Maintain consistent usage for printing pad connection info.
While this change properly calls print, consider also matching the style in other calls (e.g., x offsets). This keeps layout consistent across various calls.


136-153: Potential buffer overflow risk in printPadStatus.
Using sprintf can risk overflowing m_textBuffer. Although the current usage appears safe due to the small size (32 bytes) and short output strings, consider replacing sprintf with snprintf to ensure safety if future modifications expand the printed text.

-            sprintf(m_textBuffer, "%02X ", input.getAdc(pad, i));
+            snprintf(m_textBuffer, sizeof(m_textBuffer), "%02X ", input.getAdc(pad, i));
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro

📥 Commits

Reviewing files that changed from the base of the PR and between 0c477c3 and 0631089.

📒 Files selected for processing (2)
  • src/mips/psyqo/advancedpad.hh (1 hunks)
  • src/mips/psyqo/examples/multitap/multitap.cpp (6 hunks)
🧰 Additional context used
🪛 GitHub Check: CodeScene Cloud Delta Analysis (main)
src/mips/psyqo/examples/multitap/multitap.cpp

[warning] 156-200: ❌ New issue: Complex Method
MultitapTestScene::printPadType has a cyclomatic complexity of 13, threshold = 9. This function has many conditional statements (e.g. if, for, while), leading to lower code health. Avoid adding more conditionals and code to it without refactoring.

🔇 Additional comments (9)
src/mips/psyqo/examples/multitap/multitap.cpp (4)

35-35: Include statement seems fine.
No issues here regarding functionality or maintainability.


64-64: Method signature addition is appropriate.
The new method signature extends the class's functionality and reads coherently with the rest of the code.


215-215: Pointer notation for active pad is straightforward.
No concerns here, as it cleanly indicates which pad is selected.


227-228: Check for cohesive printing sequence.
These calls to printPadStatus and printPadType work well together. Confirm that skipping either call when the pad isn't connected remains consistent with your UI flow; you may want to conditionally print the type only if connected.

src/mips/psyqo/advancedpad.hh (5)

145-184: Convenient specialized getters for analog inputs.
These four methods (getAdc0, getAdc1, getAdc2, getAdc3) improve code readability. Validate thorough test coverage, especially for edge cases when no analog data is present.


185-207: Unified analog getter is well-designed.
The getAdc method elegantly delegates to the specialized getters and prevents boilerplate in the calling code.


209-221: getHalfword is straightforward.
The method reduces code duplication. Make sure the modulo operation is the intended approach for indexing the 2D array, as out-of-range values might silently wrap.


223-232: getPadType matches usage in multitap example.
It cleanly extracts the higher byte for the pad type. Confirm correctness of shifting logic (e.g., >> 8) for all supported devices.


234-247: Moving Command to private is good encapsulation.
Hiding the enum is beneficial unless external code requires direct access. If future external usage is anticipated, consider making it protected or adding proper accessor methods.

Comment on lines 156 to 200
void MultitapTestScene::printPadType(psyqo::AdvancedPad::Pad pad, int column, const char *name) {
auto &input = multitapTest.m_input;
char m_textBuffer[16] = {'\0'};
const auto padType = input.getPadType(pad);

print(column + 0, 11, true, name);
switch (padType) {
case psyqo::AdvancedPad::PadType::Mouse:
sprintf(m_textBuffer, "Mouse");
break;
case psyqo::AdvancedPad::PadType::NegCon:
sprintf(m_textBuffer, "NegCon");
break;
case psyqo::AdvancedPad::PadType::KonamiLightgun:
sprintf(m_textBuffer, "KonamiLightgun");
break;
case psyqo::AdvancedPad::PadType::DigitalPad:
sprintf(m_textBuffer, "DigitalPad");
break;
case psyqo::AdvancedPad::PadType::AnalogStick:
sprintf(m_textBuffer, "AnalogStick");
break;
case psyqo::AdvancedPad::PadType::NamcoLightgun:
sprintf(m_textBuffer, "NamcoLightgun");
break;
case psyqo::AdvancedPad::PadType::AnalogPad:
sprintf(m_textBuffer, "AnalogPad");
break;
case psyqo::AdvancedPad::PadType::Multitap:
sprintf(m_textBuffer, "Multitap");
break;
case psyqo::AdvancedPad::PadType::Jogcon:
sprintf(m_textBuffer, "Jogcon");
break;
case psyqo::AdvancedPad::PadType::ConfigMode:
sprintf(m_textBuffer, "ConfigMode");
break;
case psyqo::AdvancedPad::PadType::None:
sprintf(m_textBuffer, "None");
break;
default:
sprintf(m_textBuffer, "Unknown");
break;
}
print(column + 10, 11, true, m_textBuffer);
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🛠️ Refactor suggestion

Refactor to reduce cyclomatic complexity in printPadType.
This method’s complexity triggers a warning. As more pad types or logic are added, it can become unwieldy. A lookup table or map-based approach could drastically simplify this switch statement.

-switch (padType) {
  // ...
-    case psyqo::AdvancedPad::PadType::NegCon:
-        sprintf(m_textBuffer, "NegCon");
-        break;
  ...
-default:
-    sprintf(m_textBuffer, "Unknown");
-    break;
+static const struct {
+    psyqo::AdvancedPad::PadType type;
+    const char* name;
+} padTypeMap[] = {
+    { psyqo::AdvancedPad::PadType::Mouse, "Mouse" },
+    { psyqo::AdvancedPad::PadType::NegCon, "NegCon" },
+    // ...
+    { psyqo::AdvancedPad::PadType::None, "None" },
+};
+
+const auto it = std::find_if(
+    std::begin(padTypeMap), std::end(padTypeMap),
+    [padType](auto &entry) { return entry.type == padType; }
+);
+sprintf(m_textBuffer, "%s", (it != std::end(padTypeMap)) ? it->name : "Unknown");

Committable suggestion skipped: line range outside the PR's diff.

🧰 Tools
🪛 GitHub Check: CodeScene Cloud Delta Analysis (main)

[warning] 156-200: ❌ New issue: Complex Method
MultitapTestScene::printPadType has a cyclomatic complexity of 13, threshold = 9. This function has many conditional statements (e.g. if, for, while), leading to lower code health. Avoid adding more conditionals and code to it without refactoring.

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 0

🧹 Nitpick comments (5)
src/mips/psyqo/advancedpad.hh (3)

145-184: Enhance documentation for ADC getter methods.

The documentation for the ADC getter methods could be more helpful by including:

  • Value ranges (e.g., 0-255)
  • Typical use cases (e.g., analog stick position, pressure sensitivity)
  • Mapping of values to physical controls for different pad types

185-207: Consider adding bounds checking for the index parameter.

The method silently returns 0 for invalid indices, which could mask errors. Consider:

  1. Adding a precondition check for the index parameter
  2. Throwing an exception or returning an optional/error code for invalid indices

209-221: Add safety notes to getHalfword documentation.

Since this method provides raw access to pad data, the documentation should:

  • Warn about the low-level nature of the method
  • Explain the data format for each index
  • Recommend using the higher-level methods when possible
src/mips/psyqo/examples/multitap/multitap.cpp (2)

101-111: Consider moving color constants to class scope.

The WHITE and GRAY color constants are defined as static within the method. Consider moving them to class scope to:

  • Avoid recreation on each call
  • Allow reuse in other methods
  • Make them configurable if needed

113-122: LGTM! Consider moving pad names to class scope.

The implementation is clean and efficient. Consider moving the padNames array to class scope to:

  • Make it configurable
  • Avoid recreation on each call
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro

📥 Commits

Reviewing files that changed from the base of the PR and between 0631089 and 929e76f.

📒 Files selected for processing (2)
  • src/mips/psyqo/advancedpad.hh (2 hunks)
  • src/mips/psyqo/examples/multitap/multitap.cpp (6 hunks)
🧰 Additional context used
🪛 GitHub Check: CodeScene Cloud Delta Analysis (main)
src/mips/psyqo/examples/multitap/multitap.cpp

[warning] 163-206: ❌ New issue: Complex Method
MultitapTestScene::printPadType has a cyclomatic complexity of 13, threshold = 9. This function has many conditional statements (e.g. if, for, while), leading to lower code health. Avoid adding more conditionals and code to it without refactoring.

⏰ Context from checks skipped due to timeout of 90000ms (10)
  • GitHub Check: macos-arm-build-and-test
  • GitHub Check: macos-intel-build-and-test
  • GitHub Check: Codacy Static Code Analysis
  • GitHub Check: macos-build-and-test-toolchain
  • GitHub Check: asan
  • GitHub Check: aur-build
  • GitHub Check: build
  • GitHub Check: coverage
  • GitHub Check: toolchain
  • GitHub Check: cross-arm64
🔇 Additional comments (3)
src/mips/psyqo/advancedpad.hh (1)

234-247: LGTM! Good encapsulation of implementation details.

Moving the Command enum to private section is a good practice as these are internal implementation details.

src/mips/psyqo/examples/multitap/multitap.cpp (2)

163-206: Refactor suggestion from previous review still applies.

The method's cyclomatic complexity is high. The previous review's suggestion to use a lookup table would improve maintainability.

🧰 Tools
🪛 GitHub Check: CodeScene Cloud Delta Analysis (main)

[warning] 163-206: ❌ New issue: Complex Method
MultitapTestScene::printPadType has a cyclomatic complexity of 13, threshold = 9. This function has many conditional statements (e.g. if, for, while), leading to lower code health. Avoid adding more conditionals and code to it without refactoring.


220-222: LGTM! Clean and well-organized implementation.

The frame method effectively uses the new helper methods, providing a clear separation of concerns.

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 1

📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro

📥 Commits

Reviewing files that changed from the base of the PR and between 929e76f and e73f7fd.

📒 Files selected for processing (2)
  • src/mips/psyqo/advancedpad.hh (2 hunks)
  • src/mips/psyqo/examples/multitap/multitap.cpp (6 hunks)
🧰 Additional context used
🪛 GitHub Check: CodeScene Cloud Delta Analysis (main)
src/mips/psyqo/examples/multitap/multitap.cpp

[warning] 163-206: ❌ New issue: Complex Method
MultitapTestScene::printPadType has a cyclomatic complexity of 13, threshold = 9. This function has many conditional statements (e.g. if, for, while), leading to lower code health. Avoid adding more conditionals and code to it without refactoring.

⏰ Context from checks skipped due to timeout of 90000ms (10)
  • GitHub Check: macos-intel-build-and-test
  • GitHub Check: macos-arm-build-and-test
  • GitHub Check: toolchain
  • GitHub Check: macos-build-and-test-toolchain
  • GitHub Check: build
  • GitHub Check: Codacy Static Code Analysis
  • GitHub Check: coverage
  • GitHub Check: cross-arm64
  • GitHub Check: aur-build
  • GitHub Check: asan
🔇 Additional comments (6)
src/mips/psyqo/advancedpad.hh (2)

145-185: Well-documented and efficiently implemented analog input getters!

The implementation provides clear documentation about the meaning of values (e.g., "00h=Left, 80h=Center, FFh=Right") and uses efficient bit manipulation to extract the values.


242-255: Good encapsulation of low-level commands!

Moving the Command enum to private section improves encapsulation by hiding implementation details.

src/mips/psyqo/examples/multitap/multitap.cpp (4)

105-112: Clean implementation of variadic print method!

The implementation correctly handles variable arguments with proper va_start/va_end usage.


219-222: Clean and focused frame implementation!

The method has a clear separation of concerns, delegating to specific print methods.


163-206: 🛠️ Refactor suggestion

Refactor to reduce complexity and prevent potential issues.

The method has high cyclomatic complexity and potential safety issues:

  1. Large switch statement increases complexity
  2. padTypeStr could theoretically be used uninitialized if switch falls through

Consider using a static lookup table:

+    static const struct {
+        PadType type;
+        const char* name;
+    } padTypeMap[] = {
+        { PadType::Mouse, "Mouse" },
+        { PadType::NegCon, "NegCon" },
+        { PadType::KonamiLightgun, "KonamiLightgun" },
+        { PadType::DigitalPad, "DigitalPad" },
+        { PadType::AnalogStick, "AnalogStick" },
+        { PadType::NamcoLightgun, "NamcoLightgun" },
+        { PadType::AnalogPad, "AnalogPad" },
+        { PadType::Multitap, "Multitap" },
+        { PadType::Jogcon, "Jogcon" },
+        { PadType::ConfigMode, "ConfigMode" },
+        { PadType::None, "None" },
+    };
+
+    const auto it = std::find_if(std::begin(padTypeMap), std::end(padTypeMap),
+        [padType](const auto& entry) { return entry.type == padType; });
+    
+    const char* padTypeStr = (it != std::end(padTypeMap)) ? it->name : "Unknown";

Likely invalid or redundant comment.

🧰 Tools
🪛 GitHub Check: CodeScene Cloud Delta Analysis (main)

[warning] 163-206: ❌ New issue: Complex Method
MultitapTestScene::printPadType has a cyclomatic complexity of 13, threshold = 9. This function has many conditional statements (e.g. if, for, while), leading to lower code health. Avoid adding more conditionals and code to it without refactoring.


125-161: Add bounds checking for adcBytes calculation.

The calculation of adcBytes from halfWords lacks bounds checking. While unlikely, corrupted pad data could lead to buffer overflow.

Consider adding explicit bounds check:

-    const auto adcBytes = (halfWords - 1) * 2;
+    const auto adcBytes = std::min((halfWords - 1) * 2, 8u);  // Limit to maximum possible ADC bytes

Comment on lines 187 to 210
/**
* @brief Returns the state of an Analog Input.
*
* @details See the specific Analog Input functions for details.
* The index is modulo 4, so it will wrap around if it is greater than 3.
*
* @param pad The pad to query.
* @param index The index of the Analog Input.
* @return The state of the Analog Input as an unsigned 8-bit value(0-255).
*/
uint8_t getAdc(Pad pad, unsigned int index) const {
switch (index) {
case 0:
return getAdc0(pad);
case 1:
return getAdc1(pad);
case 2:
return getAdc2(pad);
case 3:
return getAdc3(pad);
default:
return 0;
}
}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

⚠️ Potential issue

Fix documentation or implementation mismatch for index wrapping.

The documentation states "The index is modulo 4, so it will wrap around if it is greater than 3", but the implementation returns 0 for indices > 3 instead of wrapping.

Consider one of these fixes:

-    switch (index) {
+    switch (index % 4) {

or update the documentation to reflect the current behavior:

-     * The index is modulo 4, so it will wrap around if it is greater than 3.
+     * Indices greater than 3 will return 0.
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
/**
* @brief Returns the state of an Analog Input.
*
* @details See the specific Analog Input functions for details.
* The index is modulo 4, so it will wrap around if it is greater than 3.
*
* @param pad The pad to query.
* @param index The index of the Analog Input.
* @return The state of the Analog Input as an unsigned 8-bit value(0-255).
*/
uint8_t getAdc(Pad pad, unsigned int index) const {
switch (index) {
case 0:
return getAdc0(pad);
case 1:
return getAdc1(pad);
case 2:
return getAdc2(pad);
case 3:
return getAdc3(pad);
default:
return 0;
}
}
/**
* @brief Returns the state of an Analog Input.
*
* @details See the specific Analog Input functions for details.
* The index is modulo 4, so it will wrap around if it is greater than 3.
*
* @param pad The pad to query.
* @param index The index of the Analog Input.
* @return The state of the Analog Input as an unsigned 8-bit value(0-255).
*/
uint8_t getAdc(Pad pad, unsigned int index) const {
switch (index % 4) {
case 0:
return getAdc0(pad);
case 1:
return getAdc1(pad);
case 2:
return getAdc2(pad);
case 3:
return getAdc3(pad);
default:
return 0;
}
}

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 0

♻️ Duplicate comments (1)
src/mips/psyqo/advancedpad.hh (1)

212-229: ⚠️ Potential issue

Fix documentation or implementation mismatch for index wrapping.

The documentation states "The index is modulo 4, so it will wrap around if it is greater than 3", but the implementation returns undefined behavior for indices > 3 due to direct array access.

Consider adding bounds checking:

-    uint16_t getHalfword(Pad pad, unsigned int index) const { return m_padData[pad][index % 4]; }
+    uint16_t getHalfword(Pad pad, unsigned int index) const {
+        const auto safeIndex = index % 4;
+        return m_padData[pad][safeIndex];
+    }
🧹 Nitpick comments (2)
src/mips/psyqo/advancedpad.hh (2)

197-210: Consider using array indexing for better maintainability.

The switch case could be simplified using array indexing, which would be more maintainable and potentially more efficient.

-    uint8_t getAdc(Pad pad, unsigned int index) const {
-        switch (index) {
-            case 0:
-                return getAdc0(pad);
-            case 1:
-                return getAdc1(pad);
-            case 2:
-                return getAdc2(pad);
-            case 3:
-                return getAdc3(pad);
-            default:
-                return 0;
-        }
-    }
+    uint8_t getAdc(Pad pad, unsigned int index) const {
+        static constexpr auto getters[] = {
+            &AdvancedPad::getAdc0,
+            &AdvancedPad::getAdc1,
+            &AdvancedPad::getAdc2,
+            &AdvancedPad::getAdc3
+        };
+        return index < 4 ? (this->*getters[index])(pad) : 0;
+    }

231-239: Enhance documentation for getPadType method.

The documentation could be more helpful by:

  1. Referencing the PadType enum values that could be returned
  2. Explaining how this relates to isPadConnected
  3. Noting any special cases or invalid states
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro

📥 Commits

Reviewing files that changed from the base of the PR and between e73f7fd and af630c4.

📒 Files selected for processing (1)
  • src/mips/psyqo/advancedpad.hh (2 hunks)
⏰ Context from checks skipped due to timeout of 90000ms (3)
  • GitHub Check: toolchain
  • GitHub Check: cross-arm64
  • GitHub Check: coverage
🔇 Additional comments (2)
src/mips/psyqo/advancedpad.hh (2)

5-5: LGTM!

Copyright year correctly updated to 2025.


242-255: LGTM! Good encapsulation of implementation details.

Moving the Command enum to private scope is a good practice as these are internal implementation details that shouldn't be exposed to users of the class.

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 0

🧹 Nitpick comments (1)
src/mips/psyqo/advancedpad.hh (1)

147-187: Consider optimizing the ADC access methods.

The implementation is correct and well-documented. However, consider these potential optimizations:

  1. The switch statement could be replaced with array indexing for better performance
  2. The individual getAdc0-3 methods could be implemented in terms of getAdc for better maintainability
-    uint8_t getAdc(Pad pad, unsigned int index) const {
-        switch (index) {
-            case 0:
-                return getAdc0(pad);
-            case 1:
-                return getAdc1(pad);
-            case 2:
-                return getAdc2(pad);
-            case 3:
-                return getAdc3(pad);
-            default:
-                return 0;
-        }
-    }
+    uint8_t getAdc(Pad pad, unsigned int index) const {
+        static const auto getters[] = {
+            &AdvancedPad::getAdc0,
+            &AdvancedPad::getAdc1,
+            &AdvancedPad::getAdc2,
+            &AdvancedPad::getAdc3
+        };
+        return index < 4 ? (this->*getters[index])(pad) : 0;
+    }

Also applies to: 189-212

📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro

📥 Commits

Reviewing files that changed from the base of the PR and between af630c4 and f3ee66e.

📒 Files selected for processing (2)
  • src/mips/psyqo/advancedpad.hh (4 hunks)
  • src/mips/psyqo/src/advancedpad.cpp (4 hunks)
⏰ Context from checks skipped due to timeout of 90000ms (9)
  • GitHub Check: Codacy Static Code Analysis
  • GitHub Check: build-openbios
  • GitHub Check: macos-build-and-test-toolchain
  • GitHub Check: aur-build
  • GitHub Check: coverage
  • GitHub Check: cross-arm64
  • GitHub Check: asan
  • GitHub Check: build
  • GitHub Check: toolchain
🔇 Additional comments (7)
src/mips/psyqo/src/advancedpad.cpp (2)

117-117: LGTM! Consistent enum class handling.

All array accesses properly use static_cast<unsigned> for the enum class values.

Also applies to: 123-123, 127-128, 138-138


51-58: LGTM! Proper use of scoped enum.

The pad processing correctly uses the new Pad:: scope for enum values.

src/mips/psyqo/advancedpad.hh (5)

5-5: LGTM!

Copyright year correctly updated to 2025.


45-45: Great improvements to type safety and encapsulation!

  • Converting Pad to a scoped enum class with explicit unsigned type improves type safety
  • Moving Command enum to private section properly encapsulates implementation details

Also applies to: 246-259


131-131: LGTM!

Methods correctly updated to handle the new enum class type using static_cast.

Also applies to: 143-145


233-243: LGTM!

The getPadType method correctly extracts and returns the pad type from the raw data.


214-231: Verify the index wrapping behavior.

The implementation now correctly matches the documentation using the modulo operator. However, let's verify that this change doesn't break any existing code.

✅ Verification successful

Index wrapping implementation verified as safe and correct

The modulo operator implementation matches the documented behavior exactly, and no existing code in the codebase uses this method, making this change completely safe.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Search for calls to getHalfword with index parameter
rg -p "getHalfword.*\([^,]+,\s*(\d+|[^)]+)\)" --type cpp

Length of output: 213


Script:

#!/bin/bash
# Search for any derived classes
ast-grep --pattern 'class $_ : $$$AdvancedPad$$$'

# Search for test files
fd -e cpp -e cc test
fd -e cpp -e cc -e hh -e h mock

# Search for any indirect usage patterns
rg -p "getHalfword" --type cpp -A 2 -B 2

Length of output: 3634


Script:

#!/bin/bash
# Check the padtest example
cat src/mips/psyqo/examples/padtest/padtest.cpp

Length of output: 6689

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 0

🧹 Nitpick comments (1)
src/mips/psyqo/advancedpad.hh (1)

205-205: Return Enum Type for getPadType

To enhance type safety and clarity, consider returning PadType instead of uint8_t in getPadType.

Apply this diff to update the return type:

-        uint8_t getPadType(Pad pad) const { return m_padData[static_cast<unsigned>(pad)][0] >> 8; }
+        PadType getPadType(Pad pad) const { return static_cast<PadType>(m_padData[static_cast<unsigned>(pad)][0] >> 8); }

Ensure that all callers handle the PadType enum appropriately.

📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro

📥 Commits

Reviewing files that changed from the base of the PR and between f3ee66e and 131796e.

📒 Files selected for processing (1)
  • src/mips/psyqo/advancedpad.hh (4 hunks)
⏰ Context from checks skipped due to timeout of 90000ms (9)
  • GitHub Check: macos-intel-build-and-test
  • GitHub Check: Codacy Static Code Analysis
  • GitHub Check: build
  • GitHub Check: asan
  • GitHub Check: toolchain
  • GitHub Check: aur-build
  • GitHub Check: cross-arm64
  • GitHub Check: coverage
  • GitHub Check: macos-build-and-test-toolchain
🔇 Additional comments (6)
src/mips/psyqo/advancedpad.hh (6)

45-45: Improved Type Safety with Scoped Enum

Changing Pad to a scoped enum class enhances type safety by preventing implicit conversions. This is a good practice.


131-131: Ensure Consistent Use of static_cast

Updating isPadConnected to use static_cast<unsigned>(pad) ensures compatibility with the new Pad enum class. Good job maintaining consistency.


185-186: Clarify Documentation Regarding Index Wrapping

The comment states, "The index is modulo 4, so it will wrap around if it is greater than 3." However, in getAdc, indices greater than 3 return 0 instead of wrapping.

Ensure the documentation accurately reflects the implementation in both methods for consistency.


208-221: Visibility Change of Command Enum

Moving the Command enum to the private section restricts its access, which is appropriate if it's not intended for public use.


161-174: 🛠️ Refactor suggestion

Consistency in Handling Invalid Indices in getAdc

The getAdc method returns 0 for indices greater than 3, while getHalfword wraps the index using modulo 4. This inconsistency might confuse users.

Consider aligning the behavior by applying the modulo operation to the index in getAdc:

-        switch (index) {
+        switch (index % 4) {

Alternatively, update the documentation to reflect that indices greater than 3 will return 0.

Likely invalid or redundant comment.


143-145: Verify Button Indexing Logic

Ensure that the button indexing in isButtonPressed correctly maps to the Button enum values. Verify that each button corresponds to the intended bit position.

Run the following script to check the correctness of button bit positions:

✅ Verification successful

Button Indexing Implementation is Correct

The button indexing logic in isButtonPressed correctly maps to the Button enum values. Each button value (0-15) corresponds to its intended bit position in the 16-bit pad data, with proper active-low signal handling.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Verify that Button enum values correspond to correct bit positions.
# Expected: Button values should range from 0 to 15 and map to the appropriate bits.

# Extract Button enum values
ast-grep --pattern $'enum Button {\n  $$$\n};' \
   | grep -oP '\w+\s*=\s*\d+' \
   | awk -F'=' '{print $1, $2}' \
   | while read -r button value; do
       bit=$((1 << value))
       printf "%s (value: %s) should correspond to bit: 0x%X\n" "$button" "$value" "$bit"
     done

Length of output: 122


Script:

#!/bin/bash
# Let's first get the Button enum declaration with context
echo "=== Button enum declaration ==="
rg -U "enum\s+(?:class\s+)?Button" src/mips/psyqo/advancedpad.hh -A 20

echo -e "\n=== Pad data structure and related definitions ==="
rg "m_padData" src/mips/psyqo/advancedpad.hh -B 2 -A 2

echo -e "\n=== Any documentation about button mapping ==="
rg -B 2 -A 2 "Button" src/mips/psyqo/advancedpad.hh

Length of output: 2950

@johnbaumann
Copy link
Collaborator Author

Actionable comments posted: 0

🧹 Nitpick comments (1)

src/mips/psyqo/advancedpad.hh (1)> 205-205: Return Enum Type for getPadType

To enhance type safety and clarity, consider returning PadType instead of uint8_t in getPadType.
Apply this diff to update the return type:

-        uint8_t getPadType(Pad pad) const { return m_padData[static_cast<unsigned>(pad)][0] >> 8; }
+        PadType getPadType(Pad pad) const { return static_cast<PadType>(m_padData[static_cast<unsigned>(pad)][0] >> 8); }

Ensure that all callers handle the PadType enum appropriately.

📜 Review details

uint8_t makes more sense here, because the function could return unknown and undocumented PadType values. Silly wabbit!

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 0

🧹 Nitpick comments (1)
src/mips/psyqo/advancedpad.hh (1)

161-176: Consider simplifying the index selection logic.

The switch statement could be simplified using array indexing for better maintainability.

-        switch (index) {
-            case 0:
-                return m_padData[padIndex].adc[0];
-            case 1:
-                return m_padData[padIndex].adc[1];
-            case 2:
-                return m_padData[padIndex].adc[2];
-            case 3:
-                return m_padData[padIndex].adc[3];
-            default:
-                return 0;
-        }
+        return index <= 3 ? m_padData[padIndex].adc[index] : 0;
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro

📥 Commits

Reviewing files that changed from the base of the PR and between 131796e and 2e21c2f.

📒 Files selected for processing (2)
  • src/mips/psyqo/advancedpad.hh (5 hunks)
  • src/mips/psyqo/src/advancedpad.cpp (6 hunks)
🚧 Files skipped from review as they are similar to previous changes (1)
  • src/mips/psyqo/src/advancedpad.cpp
⏰ Context from checks skipped due to timeout of 90000ms (10)
  • GitHub Check: macos-arm-build-and-test
  • GitHub Check: macos-intel-build-and-test
  • GitHub Check: coverage
  • GitHub Check: macos-build-and-test-toolchain
  • GitHub Check: asan
  • GitHub Check: aur-build
  • GitHub Check: build
  • GitHub Check: cross-arm64
  • GitHub Check: toolchain
  • GitHub Check: Codacy Static Code Analysis
🔇 Additional comments (4)
src/mips/psyqo/advancedpad.hh (4)

5-5: LGTM: Copyright year update is appropriate.


45-45: Well-structured enum improvements!

Good changes:

  • Converting Pad to a scoped enum class improves type safety
  • Explicit unsigned type prevents potential sign issues
  • Moving Command to private section properly encapsulates implementation details

Also applies to: 212-225


209-209: Appropriate use of uint8_t for getPadType.

Good decision to use uint8_t instead of PadType enum as the return type, as it allows handling unknown pad types that might not be documented in the enum.


240-248: Well-designed data structure!

The PadData union effectively combines:

  • Structured access through named fields
  • Raw access through packed array
  • Efficient memory layout

Also applies to: 250-250

Copy link
Contributor

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 0

🧹 Nitpick comments (2)
src/mips/psyqo/advancedpad.hh (2)

188-198: Consider using PadType enum with unknown value handling.

While returning uint8_t allows for handling unknown pad types, consider using the PadType enum with an additional Unknown value to maintain type safety while supporting undocumented values:

-    uint8_t getPadType(Pad pad) const { return m_padData[static_cast<unsigned>(pad)].padType; }
+    PadType getPadType(Pad pad) const {
+        uint8_t type = m_padData[static_cast<unsigned>(pad)].padType;
+        switch (type) {
+            case Mouse: return PadType::Mouse;
+            case NegCon: return PadType::NegCon;
+            // ... other known types ...
+            default: return PadType::Unknown;
+        }
+    }

Line range hint 251-274: Consider adding a named constant for clarity.

The bitwise AND with 7 ensures the pad index stays within range, but consider adding a named constant to make the code more self-documenting:

+    static constexpr unsigned PAD_INDEX_MASK = 0x7;  // 8 possible pad values (0-7)

     // operators implementation
-    return pad = static_cast<psyqo::AdvancedPad::Pad>((static_cast<unsigned>(pad) + 1) & 7);
+    return pad = static_cast<psyqo::AdvancedPad::Pad>((static_cast<unsigned>(pad) + 1) & PAD_INDEX_MASK);
📜 Review details

Configuration used: CodeRabbit UI
Review profile: CHILL
Plan: Pro

📥 Commits

Reviewing files that changed from the base of the PR and between 2e21c2f and 3215462.

📒 Files selected for processing (1)
  • src/mips/psyqo/advancedpad.hh (5 hunks)
⏰ Context from checks skipped due to timeout of 90000ms (8)
  • GitHub Check: macos-arm-build-and-test
  • GitHub Check: coverage
  • GitHub Check: cross-arm64
  • GitHub Check: build
  • GitHub Check: toolchain
  • GitHub Check: aur-build
  • GitHub Check: asan
  • GitHub Check: macos-build-and-test-toolchain
🔇 Additional comments (6)
src/mips/psyqo/advancedpad.hh (6)

5-5: LGTM: Copyright year is up to date.


45-45: Great improvement: Enhanced type safety with enum class.

The conversion to enum class with explicit unsigned type prevents accidental implicit conversions and makes the code more type-safe. The explicit static_cast usage throughout the code properly handles this change.


147-165: LGTM: Well-implemented analog input retrieval.

The getAdc method is well-documented and includes proper bounds checking for the index parameter.


167-186: Fix documentation or implementation mismatch for index wrapping.

The documentation states "The index is modulo 4, so it will wrap around if it is greater than 3", which matches the implementation using index % 4. However, this is inconsistent with the behavior of getAdc which returns 0 for indices > 3.


201-214: LGTM: Improved encapsulation of command constants.

Moving the Command enum to private scope is a good practice as these are implementation details not needed by clients of the class.


229-237: LGTM: Well-designed data structure.

The PadData union elegantly provides both structured access to individual fields and raw access to packed data, which is useful for different access patterns while maintaining memory efficiency.

Copy link

codecov bot commented Jan 10, 2025

Codecov Report

All modified and coverable lines are covered by tests ✅

Project coverage is 9.25%. Comparing base (e6f816d) to head (ceea701).
Report is 4 commits behind head on main.

Additional details and impacted files
@@            Coverage Diff             @@
##             main    #1815      +/-   ##
==========================================
- Coverage    9.26%    9.25%   -0.01%     
==========================================
  Files         468      468              
  Lines      144113   144115       +2     
==========================================
  Hits        13345    13345              
- Misses     130768   130770       +2     

☔ View full report in Codecov by Sentry.
📢 Have feedback on the report? Share it here.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants