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

Adding GNU C++ demangler #1834

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

Conversation

nicolasnoble
Copy link
Member

No description provided.

Copy link
Contributor

coderabbitai bot commented Jan 7, 2025

Walkthrough

The pull request introduces the PEGTL (Parsing Expression Grammar Template Library) as a new third-party dependency for the PCSX-Redux project. This involves adding a Git submodule for PEGTL, updating build configurations across different platforms (Makefile and Visual Studio), and implementing a new GNU C++ name demangler using the PEGTL library. The changes include modifying project files to include the new library's headers, adding a new source and header file for the demangler, and updating the project's license documentation.

Changes

File Change Summary
.gitmodules Added new submodule for PEGTL at third_party/PEGTL
LICENSES.md Removed PEGTL library entry
Makefile Added include path for PEGTL headers
src/support/gnu-c++-demangler.h New header file declaring demangler functions
src/support/gnu-c++-demangler.cc New source file implementing GNU C++ demangler
third_party/PEGTL Added subproject commit for PEGTL library
vsprojects/common.props Added PEGTL include directory
vsprojects/pcsx-redux.sln Added new project entries for PEGTL
vsprojects/support/support.vcxproj Added new source and header files for demangler
vsprojects/support/support.vcxproj.filters Updated project filters for new files
compile_flags.txt Updated C++ standard and include paths
tests/support/gnu-c++-demangler.cc New test file for GNUDemangler functionality
vsprojects/tests/support/testsupport.vcxproj Included new test source file in project

Sequence Diagram

sequenceDiagram
    participant Caller
    participant Demangler
    participant PEGTL
    
    Caller->>Demangler: demangle(mangled_name)
    Demangler->>PEGTL: Parse mangled name
    PEGTL-->>Demangler: Parsing result
    alt Parsing Successful
        Demangler-->>Caller: Demangled name
    else Parsing Failed
        Demangler-->>Caller: Original mangled name
    end
Loading

Poem

🐰 A Rabbit's Ode to Demangling
With PEGTL's magic, names unwind,
Mangled symbols now refined,
C++ secrets, once obscure,
Now decoded, crystal pure!
Hop along, code detective mine! 🕵️‍♀️


🪧 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: 2

🧹 Nitpick comments (3)
src/support/gnu-c++-demangler.cc (1)

366-371: Enhance error handling in the demangle function.

Currently, if parsing fails, the function returns the original mangled name without any indication of the error. Consider logging the error or providing more detailed error information to assist with debugging.

src/support/gnu-c++-demangler.h (1)

36-38: Add documentation for public functions.

The functions internalCheck(), trace(std::string_view mangled), and demangle(std::string_view mangled) are part of the public interface. Adding documentation comments will improve readability and help other developers understand the purpose and usage of these functions.

LICENSES.md (1)

56-56: Fix markdown list indentation.

The list item indentation is inconsistent with the rest of the file. Consider fixing the indentation to match other entries.

- - [PEGTL](https://github.com/taocpp/PEGTL)
+ - [PEGTL](https://github.com/taocpp/PEGTL)
🧰 Tools
🪛 Markdownlint (0.37.0)

56-56: Expected: 0; Actual: 1
Unordered list indentation

(MD007, ul-indent)

📜 Review details

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

📥 Commits

Reviewing files that changed from the base of the PR and between e6f816d and 1e1cc98.

📒 Files selected for processing (10)
  • .gitmodules (1 hunks)
  • LICENSES.md (1 hunks)
  • Makefile (1 hunks)
  • src/support/gnu-c++-demangler.cc (1 hunks)
  • src/support/gnu-c++-demangler.h (1 hunks)
  • third_party/PEGTL (1 hunks)
  • vsprojects/common.props (1 hunks)
  • vsprojects/pcsx-redux.sln (2 hunks)
  • vsprojects/support/support.vcxproj (2 hunks)
  • vsprojects/support/support.vcxproj.filters (2 hunks)
✅ Files skipped from review due to trivial changes (1)
  • third_party/PEGTL
👮 Files not reviewed due to content moderation or server errors (1)
  • vsprojects/support/support.vcxproj.filters
🧰 Additional context used
🪛 Markdownlint (0.37.0)
LICENSES.md

56-56: Expected: 0; Actual: 1
Unordered list indentation

(MD007, ul-indent)

⏰ Context from checks skipped due to timeout of 90000ms (1)
  • GitHub Check: Codacy Static Code Analysis
🔇 Additional comments (8)
vsprojects/common.props (1)

11-11: Verify the correctness of the added include directory path.

The path $(SolutionDir)..\third_party\PEGTL\include; has been added to AdditionalIncludeDirectories. Please verify that this path is correct and accessible during the build process to ensure that the PEGTL headers are included properly.

Makefile (1)

39-39: LGTM!

The PEGTL include path is correctly added and follows the existing pattern in the Makefile.

.gitmodules (1)

97-99: LGTM! Clean addition of PEGTL dependency

The addition of PEGTL as a submodule follows project conventions and uses the official repository.

vsprojects/support/support.vcxproj (2)

163-163: LGTM! Header file addition follows conventions

The gnu-c++-demangler.h header is properly integrated into the project structure.


213-215: Verify if /bigobj is needed for other configurations

The /bigobj compiler option is added only for Debug|x64. Consider if this option is also needed for other configurations to handle large object files generated by template-heavy PEGTL code.

✅ Verification successful

Keep /bigobj option specific to Debug|x64 configuration

The current configuration appears correct. The gnu-c++-demangler.cc file (20KB) is the largest in the support directory and likely generates large object files specifically in Debug builds due to template expansion and debug symbols. Release builds typically have better template instantiation optimization, explaining why the option isn't needed there.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Check if other configurations might need /bigobj by looking for similar patterns
rg -l "bigobj" vsprojects/

Length of output: 61


Script:

#!/bin/bash
# Get more context around bigobj usage
rg -C 5 "bigobj" vsprojects/support/support.vcxproj

# Check for the specific file in other configurations
rg -A 5 "gnu-c\+\+-demangler\.cc" vsprojects/support/support.vcxproj

# Look for other potential template-heavy files
fd . src/support -e cc -x ls -l {} \;

Length of output: 2767

vsprojects/pcsx-redux.sln (3)

110-114: LGTM! Well-structured PEGTL integration

The PEGTL root folder is properly added under third_party with the main header.


780-785: LGTM! Proper nesting of PEGTL components

The folder hierarchy correctly organizes PEGTL components (core, contrib, internal) maintaining clean project structure.


789-789: Update SolutionGuid to maintain solution identity

The solution GUID has been updated, which is expected when adding new projects.

Comment on lines +103 to +104
size *= 10;
size += in.peek_char() - '0';
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

Check for potential integer overflow in size calculation.

The size variable is calculated by accumulating digit characters in lines 103-104. If the mangled name contains a very large number, there is a risk of integer overflow. Consider adding a check to prevent overflow or using a larger integer type like std::size_t.

Comment on lines +102 to +105
while (in.size() > 0 && std::isdigit(in.peek_char())) {
size *= 10;
size += in.peek_char() - '0';
in.bump(1);
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

Cast char to unsigned char when using std::isdigit.

In the source_name::match function, std::isdigit(in.peek_char()) is called on line 102. Since std::isdigit expects an int that is representable as an unsigned char or EOF, passing a char directly can lead to undefined behavior if char is signed and contains a negative value. To ensure correct behavior, cast the character to unsigned char before passing it to std::isdigit.

Apply this diff to fix the issue:

-            while (in.size() > 0 && std::isdigit(in.peek_char())) {
+            while (in.size() > 0 && std::isdigit(static_cast<unsigned char>(in.peek_char()))) {
📝 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
while (in.size() > 0 && std::isdigit(in.peek_char())) {
size *= 10;
size += in.peek_char() - '0';
in.bump(1);
while (in.size() > 0 && std::isdigit(static_cast<unsigned char>(in.peek_char()))) {
size *= 10;
size += in.peek_char() - '0';
in.bump(1);

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: 5

🧹 Nitpick comments (1)
src/support/gnu-c++-demangler.cc (1)

50-55: Improve debugbreak implementation for non-Windows platforms.

The non-Windows implementation of debugbreak() is empty. Consider:

  1. Using platform-specific debug break mechanisms (e.g., raise(SIGTRAP) on Unix).
  2. Adding a compile-time warning when debug breaks are unavailable.
#ifdef _WIN32
void debugbreak() {
    if (IsDebuggerPresent()) __debugbreak();
}
#else
void debugbreak() {
    #if defined(__unix__) || defined(__unix) || defined(__APPLE__)
        raise(SIGTRAP);
    #else
        #warning "Debug break not implemented for this platform"
    #endif
}
#endif
📜 Review details

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

📥 Commits

Reviewing files that changed from the base of the PR and between 1e1cc98 and f433a31.

📒 Files selected for processing (2)
  • src/support/gnu-c++-demangler.cc (1 hunks)
  • src/support/gnu-c++-demangler.h (1 hunks)
⏰ 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: build
  • GitHub Check: toolchain
  • GitHub Check: asan
  • GitHub Check: aur-build
  • GitHub Check: macos-build-and-test-toolchain
  • GitHub Check: cross-arm64
  • GitHub Check: coverage
  • GitHub Check: Codacy Static Code Analysis
🔇 Additional comments (1)
src/support/gnu-c++-demangler.cc (1)

129-148: ⚠️ Potential issue

Fix potential issues in source_name implementation.

The source_name implementation has several issues:

  1. Integer overflow possible in size calculation
  2. Unsafe usage of std::isdigit
  3. No bounds checking on string_view access
 struct source_name {
     using rule_t = source_name;
     using subs_t = pegtl::nothing<void>;

     template <pegtl::apply_mode A, pegtl::rewind_mode M, template <typename...> class Action,
               template <typename...> class Control, typename ParseInput, typename... States>
     [[nodiscard]] static bool match(ParseInput& in, States&&... st) {
         auto marker = in.template mark<M>();
-        unsigned size = 0;
+        std::size_t size = 0;
+        const std::size_t maxSize = std::numeric_limits<std::size_t>::max();
         while (in.size() > 0 && std::isdigit(static_cast<unsigned char>(in.peek_char()))) {
+            if (size > maxSize / 10) return marker(false);  // Prevent overflow
             size *= 10;
-            size += in.peek_char() - '0';
+            std::size_t digit = in.peek_char() - '0';
+            if (size > maxSize - digit) return marker(false);  // Prevent overflow
+            size += digit;
             in.bump(1);
         }
         if (size == 0) return marker(false);
         if (in.size() < size) return marker(false);
         in.bump(size);
         return marker(true);
     }
 };

Likely invalid or redundant comment.

Comment on lines 36 to 39
bool internalCheck();
bool trace(std::string_view mangled);
bool printDot(std::string_view mangled);
std::string demangle(std::string_view mangled);
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

Add documentation for the public API.

The public functions lack documentation. Please add Doxygen-style comments explaining:

  • Purpose of each function
  • Parameters and their meaning
  • Return value semantics
  • Example usage if applicable

Example documentation format:

/**
 * Performs internal validation of the demangler grammar.
 * @return true if the grammar is valid, false otherwise.
 */
bool internalCheck();

/**
 * Traces the demangling process for a mangled name.
 * @param mangled The mangled name to trace
 * @return true if tracing succeeded, false otherwise
 */
bool trace(std::string_view mangled);


MIT License

Copyright (c) 2025 PCSX-Redux authors
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

Update the copyright year.

The copyright year is set to 2025, which is in the future. Please update it to the current year (2024).

-Copyright (c) 2025 PCSX-Redux authors
+Copyright (c) 2024 PCSX-Redux authors
📝 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
Copyright (c) 2025 PCSX-Redux authors
Copyright (c) 2024 PCSX-Redux authors

Comment on lines +58 to +63
struct DemanglerState {
std::string tostring() { return ""; }
std::vector<std::string> bag;
std::string_view lastName;
const std::type_info* lastAction;
};
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

Improve DemanglerState implementation.

The DemanglerState class needs improvements:

  1. tostring() returns an empty string but should format the state.
  2. Member variables lack documentation.
  3. Consider making tostring() const-qualified.
struct DemanglerState {
    /// Returns a string representation of the current state
    std::string tostring() const {
        std::string result;
        if (!bag.empty()) {
            result = "Names: ";
            for (const auto& name : bag) {
                result += name + ", ";
            }
            result.resize(result.size() - 2);  // Remove trailing ", "
        }
        return result;
    }

    std::vector<std::string> bag;      ///< Collected names during parsing
    std::string_view lastName;         ///< Most recently parsed name
    const std::type_info* lastAction;  ///< Type info of last executed action
};

Comment on lines +382 to +1870
debugbreak();
throw Unimplemented("local_name");
}
};
template <>
struct DemanglerAction<data_member_prefix> {
template <typename ActionInput>
static void apply(const ActionInput& in, DemanglerState& state) {
debugbreak();
throw Unimplemented("data_member_prefix");
}
};
template <>
struct DemanglerAction<seq_id> {
template <typename ActionInput>
static void apply(const ActionInput& in, DemanglerState& state) {
debugbreak();
throw Unimplemented("seq_id");
}
};
template <>
struct DemanglerAction<substitution_simple> {
template <typename ActionInput>
static void apply(const ActionInput& in, DemanglerState& state) {
debugbreak();
throw Unimplemented("substitution_simple");
}
};
template <>
struct DemanglerAction<substitution_std> {
template <typename ActionInput>
static void apply(const ActionInput& in, DemanglerState& state) {
debugbreak();
throw Unimplemented("substitution_std");
}
};
template <>
struct DemanglerAction<substitution_std_allocator> {
template <typename ActionInput>
static void apply(const ActionInput& in, DemanglerState& state) {
debugbreak();
throw Unimplemented("substitution_std_allocator");
}
};
template <>
struct DemanglerAction<substitution_std_basic_string> {
template <typename ActionInput>
static void apply(const ActionInput& in, DemanglerState& state) {
debugbreak();
throw Unimplemented("substitution_std_basic_string");
}
};
template <>
struct DemanglerAction<substitution_std_basic_string_full> {
template <typename ActionInput>
static void apply(const ActionInput& in, DemanglerState& state) {
debugbreak();
throw Unimplemented("substitution_std_basic_string_full");
}
};
template <>
struct DemanglerAction<substitution_std_basic_istream> {
template <typename ActionInput>
static void apply(const ActionInput& in, DemanglerState& state) {
debugbreak();
throw Unimplemented("substitution_std_basic_istream");
}
};
template <>
struct DemanglerAction<substitution_std_basic_ostream> {
template <typename ActionInput>
static void apply(const ActionInput& in, DemanglerState& state) {
debugbreak();
throw Unimplemented("substitution_std_basic_ostream");
}
};
template <>
struct DemanglerAction<substitution_std_basic_iostream> {
template <typename ActionInput>
static void apply(const ActionInput& in, DemanglerState& state) {
debugbreak();
throw Unimplemented("substitution_std_basic_iostream");
}
};
template <>
struct DemanglerAction<substitution> {
template <typename ActionInput>
static void apply(const ActionInput& in, DemanglerState& state) {
debugbreak();
throw Unimplemented("substitution");
}
};
template <>
struct DemanglerAction<std_name> {
template <typename ActionInput>
static void apply(const ActionInput& in, DemanglerState& state) {
debugbreak();
throw Unimplemented("std_name");
}
};
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

Improve error handling in DemanglerAction implementations.

The current implementation has several issues:

  1. All actions throw exceptions, making error recovery impossible
  2. No progress tracking or partial results
  3. Debug breaks in every action make debugging cumbersome

Consider implementing a more robust error handling strategy:

  1. Use a Result type instead of exceptions:
template <typename Rule>
struct DemanglerAction : pegtl::nothing<Rule> {};

// Example improved implementation
template <>
struct DemanglerAction<mangled_name> {
    template <typename ActionInput>
    static void apply(const ActionInput& in, DemanglerState& state) {
        if (state.errorOccurred) return;  // Skip if already in error state
        try {
            // Implementation here
        } catch (const std::exception& e) {
            state.errorOccurred = true;
            state.errorMessage = e.what();
            state.errorLocation = in.position();
        }
    }
};
  1. Add progress tracking:
struct DemanglerState {
    // ... existing members ...
    
    /// Tracks parsing progress
    struct Progress {
        std::size_t totalRules = 0;
        std::size_t completedRules = 0;
        std::string currentRule;
    } progress;

    /// Error state
    bool errorOccurred = false;
    std::string errorMessage;
    pegtl::position errorLocation;
};

Comment on lines +1894 to +1905
std::string PCSX::GNUDemangler::demangle(std::string_view mangled) {
try {
DemanglerState state;
pegtl::string_input in(mangled, "mangled");
auto result = pegtl::parse<mangled_name, DemanglerAction>(in, state);
if (result) state.tostring();
} catch (Unimplemented unimplemented) {
std::cout << unimplemented.name;
} catch (...) {
}
return std::string(mangled);
}
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

Improve error handling in demangle function.

The current implementation has several issues:

  1. Silently catches all exceptions
  2. Only prints unimplemented rule names
  3. Always returns the original mangled name on failure

Consider implementing proper error reporting:

struct DemangleError {
    enum class Type {
        Unimplemented,
        InvalidInput,
        ParseError,
        Internal
    };
    Type type;
    std::string message;
    std::optional<std::string> ruleName;
};

std::variant<std::string, DemangleError> demangle(std::string_view mangled) {
    try {
        DemanglerState state;
        pegtl::string_input in(mangled, "mangled");
        if (!pegtl::parse<mangled_name, DemanglerAction>(in, state)) {
            return DemangleError{
                DemangleError::Type::ParseError,
                "Failed to parse mangled name",
                std::nullopt
            };
        }
        return state.tostring();
    } catch (const Unimplemented& e) {
        return DemangleError{
            DemangleError::Type::Unimplemented,
            "Unimplemented demangling rule",
            std::string(e.name)
        };
    } catch (const std::exception& e) {
        return DemangleError{
            DemangleError::Type::Internal,
            e.what(),
            std::nullopt
        };
    }
}

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/support/gnu-c++-demangler.cc (2)

5-5: Update the copyright year.

The copyright year is set to 2025, which is in the future. Consider updating it to the current year.

-Copyright (c) 2025 PCSX-Redux authors
+Copyright (c) 2024 PCSX-Redux authors

50-55: Enhance debug break for non-Windows platforms.

The non-Windows version of debugbreak() is empty. Consider using platform-specific debug break mechanisms for better debugging support.

 #else
-void debugbreak() {}
+void debugbreak() {
+#if defined(__APPLE__)
+    __builtin_debugtrap();
+#elif defined(__unix__)
+    raise(SIGTRAP);
+#else
+    // Fall back to empty implementation
+#endif
+}
📜 Review details

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

📥 Commits

Reviewing files that changed from the base of the PR and between f433a31 and 12443a5.

📒 Files selected for processing (2)
  • src/support/gnu-c++-demangler.cc (1 hunks)
  • src/support/gnu-c++-demangler.h (1 hunks)
🚧 Files skipped from review as they are similar to previous changes (1)
  • src/support/gnu-c++-demangler.h
⏰ 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: macos-build-and-test-toolchain
  • GitHub Check: toolchain
  • GitHub Check: coverage
  • GitHub Check: aur-build
  • GitHub Check: cross-arm64
🔇 Additional comments (5)
src/support/gnu-c++-demangler.cc (5)

58-63: Improve DemanglerState implementation.

The current implementation has several issues:

  1. tostring() returns an empty string but is used in demangle()
  2. Member variables lack documentation
  3. Consider making tostring() const-qualified

138-140: Check for potential integer overflow in size calculation.

The size variable is calculated by accumulating digit characters. If the mangled name contains a very large number, there is a risk of integer overflow.


138-141: Cast char to unsigned char when using std::isdigit.

Using std::isdigit with char can lead to undefined behavior if char is signed and contains a negative value.


382-1870: Improve error handling in DemanglerAction implementations.

The current implementation has several issues:

  1. All actions throw exceptions, making error recovery impossible
  2. No progress tracking or partial results
  3. Debug breaks in every action make debugging cumbersome

1894-1905: Improve error handling in demangle function.

The current implementation has several issues:

  1. Silently catches all exceptions
  2. Only prints unimplemented rule names
  3. Always returns the original mangled name on failure

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 (5)
src/support/gnu-c++-demangler.cc (5)

61-66: 🛠️ Refactor suggestion

Improve DemanglerState implementation.

The DemanglerState class needs improvements:

  1. tostring() returns an empty string but should format the state.
  2. Member variables lack documentation.
  3. Consider making tostring() const-qualified.

Apply this diff to improve the implementation:

 struct DemanglerState {
-    std::string tostring() { return ""; }
-    std::vector<std::string> bag;
-    std::string_view lastName;
-    const std::type_info* lastAction;
+    /// Returns a string representation of the current state
+    std::string tostring() const {
+        std::string result;
+        if (!bag.empty()) {
+            result = "Names: ";
+            for (const auto& name : bag) {
+                result += name + ", ";
+            }
+            result.resize(result.size() - 2);  // Remove trailing ", "
+        }
+        return result;
+    }
+
+    std::vector<std::string> bag;      ///< Collected names during parsing
+    std::string_view lastName;         ///< Most recently parsed name
+    const std::type_info* lastAction;  ///< Type info of last executed action
 };

141-143: ⚠️ Potential issue

Check for potential integer overflow in size calculation.

The size variable is calculated by accumulating digit characters. If the mangled name contains a very large number, there is a risk of integer overflow.

Apply this diff to prevent overflow:

-unsigned size = 0;
+std::size_t size = 0;
 while (in.size() > 0 && std::isdigit(in.peek_char())) {
+    const auto digit = in.peek_char() - '0';
+    if (size > (std::numeric_limits<std::size_t>::max() - digit) / 10) {
+        return marker(false);  // Would overflow
+    }
     size *= 10;
-    size += in.peek_char() - '0';
+    size += digit;

141-141: ⚠️ Potential issue

Cast char to unsigned char when using std::isdigit.

std::isdigit expects an int that is representable as an unsigned char or EOF. Passing a char directly can lead to undefined behavior if char is signed and contains a negative value.

Apply this diff to fix the issue:

-while (in.size() > 0 && std::isdigit(in.peek_char())) {
+while (in.size() > 0 && std::isdigit(static_cast<unsigned char>(in.peek_char()))) {

385-1873: 🛠️ Refactor suggestion

Improve error handling in DemanglerAction implementations.

The current implementation has several issues:

  1. All actions throw exceptions, making error recovery impossible
  2. No progress tracking or partial results
  3. Debug breaks in every action make debugging cumbersome

Consider implementing a more robust error handling strategy as shown in the past review comments.


1897-1908: 🛠️ Refactor suggestion

Improve error handling in demangle function.

The current implementation has several issues:

  1. Silently catches all exceptions
  2. Only prints unimplemented rule names
  3. Always returns the original mangled name on failure

Consider implementing proper error reporting as shown in the past review comments.

🧹 Nitpick comments (4)
tests/support/gnu-c++-demangler.cc (2)

1-2: Update the copyright year.

The copyright year is set to 2025, which is in the future. Consider updating it to the current year.

-   Copyright (C) 2025 PCSX-Redux authors                                 *
+   Copyright (C) 2024 PCSX-Redux authors                                 *

23-72: Consider enhancing test coverage and documentation.

The test suite is comprehensive and well-structured. Consider these improvements:

  1. Add comments explaining the mangling patterns for complex cases (e.g., lines 55-57 with template arithmetic).
  2. Add negative test cases to verify proper error handling for invalid mangled names.

Would you like me to help generate additional test cases for invalid inputs and add explanatory comments for the complex test cases?

vsprojects/support/support.vcxproj (1)

213-216: Consider documenting the need for /bigobj.

The /bigobj compiler option is correctly added for the demangler, which suggests it generates many symbols. Consider:

  1. Adding a comment explaining why this option is needed
  2. Documenting this requirement in the project's build documentation

Would you like me to help document this compiler requirement in the project's README or build documentation?

src/support/gnu-c++-demangler.cc (1)

5-5: Update the copyright year.

The copyright year is set to 2025, which is in the future. Consider updating it to the current year.

-Copyright (c) 2025 PCSX-Redux authors
+Copyright (c) 2024 PCSX-Redux authors
📜 Review details

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

📥 Commits

Reviewing files that changed from the base of the PR and between 12443a5 and bfd9f4a.

📒 Files selected for processing (5)
  • compile_flags.txt (1 hunks)
  • src/support/gnu-c++-demangler.cc (1 hunks)
  • tests/support/gnu-c++-demangler.cc (1 hunks)
  • vsprojects/support/support.vcxproj (2 hunks)
  • vsprojects/tests/support/testsupport.vcxproj (1 hunks)
🧰 Additional context used
🪛 LanguageTool
compile_flags.txt

[duplication] ~4-~4: Possibile errore di battitura: parola ripetuta
Context: -std=c++2b -fexceptions -I. -Isrc -Ithird_party -Ithird_party/ELFIO -Ithird_party/fmt/include/ -Ithir...

(ITALIAN_WORD_REPEAT_RULE)

⏰ 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: cross-arm64
  • GitHub Check: Codacy Static Code Analysis
  • GitHub Check: asan
  • GitHub Check: coverage
  • GitHub Check: aur-build
  • GitHub Check: toolchain
  • GitHub Check: build
  • GitHub Check: macos-build-and-test-toolchain
🔇 Additional comments (4)
tests/support/gnu-c++-demangler.cc (1)

20-21: LGTM!

The includes are minimal and appropriate for the test file.

vsprojects/tests/support/testsupport.vcxproj (1)

83-83: LGTM!

The GNU C++ demangler test file is correctly added to the project.

compile_flags.txt (1)

1-1: Verify compiler compatibility with C++2b.

The upgrade to C++2b and addition of PEGTL include path are appropriate. However, ensure that:

  1. All development environments support C++2b features
  2. PEGTL library is compatible with C++2b

Run this script to check compiler versions across the build environment:

Also applies to: 19-19

✅ Verification successful

C++2b compatibility verified.

The codebase already makes extensive use of modern C++ features including concepts, coroutines, consteval, and ranges. Since C++2b (C++23) is a superset of C++20, and the project's CI shows successful builds across platforms, the upgrade to C++2b is appropriate and compatible with the development environment.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Description: Check compiler versions and C++2b support

# Check compiler version
c++ --version

# Verify C++2b support
echo '#include <version>' > test.cpp
echo 'int main() { return 0; }' >> test.cpp
c++ -std=c++2b -c test.cpp 2>&1 || echo "C++2b not supported"

Length of output: 344


Script:

#!/bin/bash
# Search for CI configs, build files, and setup documentation
echo "=== CI Configurations ==="
fd -e yml -e yaml . .github/workflows --exec cat {}

echo -e "\n=== Build System Files ==="
fd CMakeLists.txt -x cat {}

echo -e "\n=== Setup Documentation ==="
fd -e md . -d 1 --exec cat {}

Length of output: 73528


Script:

#!/bin/bash
# Search for compiler version requirements and C++20 feature usage
echo "=== Searching for compiler requirements ==="
rg -i "cmake_minimum|c\+\+|compiler|std=" --type cmake

echo -e "\n=== Searching for C++20 feature usage ==="
rg "concept|requires|constinit|consteval|co_await|co_yield|co_return|ranges::|views::|<span>|<concepts>|<ranges>|<coroutine>" --type cpp

Length of output: 37200

src/support/gnu-c++-demangler.cc (1)

52-58: LGTM!

The debug utilities are properly implemented with platform-specific handling.

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.

1 participant