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

libclang 19.1.7 #292

Merged
merged 2 commits into from
Mar 18, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
20 changes: 16 additions & 4 deletions dstep/translator/Context.d
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@ import dstep.translator.Output;
import dstep.translator.Translator;
import dstep.translator.TypedefIndex;
import dstep.translator.HeaderIndex;
import dstep.translator.Util;

class Context
{
Expand Down Expand Up @@ -194,7 +195,7 @@ class Context

private string translateSpellingImpl(in Cursor cursor)
{
return cursor.spelling == ""
return cursor.spelling.isUnnamed()
? generateAnonymousName(cursor)
: cursor.spelling;
}
Expand Down Expand Up @@ -272,7 +273,7 @@ class Context
{
auto typedefp = typedefParent(cursor.canonical);

if (typedefp.isValid && cursor.spelling == "")
if (typedefp.isValid && cursor.spelling.isUnnamed)
{
spelling = typedefp.spelling;
}
Expand Down Expand Up @@ -378,8 +379,19 @@ bool variablesInParentScope(Cursor cursor)
*/
bool shouldBeAnonymous(Context context, Cursor cursor)
{
return cursor.type.isAnonymous &&
context.typedefIndex.typedefParent(cursor).isEmpty;
import std.string: startsWith;

/* Unfortunately, libclang isAnonymous doesn't seem to work on enums
* https://stackoverflow.com/a/35184821
* Checking for the string value instead is brittle and might break in a future
* version, but I don't think there's a better way
*/
if (cursor.kind == CXCursorKind.enumDecl)
return cursor.spelling.isUnnamed
&& context.typedefIndex.typedefParent(cursor).isEmpty;
else
return cursor.type.isAnonymous &&
context.typedefIndex.typedefParent(cursor).isEmpty;
}

/**
Expand Down
3 changes: 2 additions & 1 deletion dstep/translator/Declaration.d
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@ import clang.Cursor;

import dstep.translator.Translator;
import dstep.translator.Output;
import dstep.translator.Util;

abstract class Declaration
{
Expand Down Expand Up @@ -44,6 +45,6 @@ abstract class Declaration
@property string spelling ()
{
auto name = cursor.spelling;
return name.length || parent.isEmpty ? name : translator.context.generateAnonymousName(cursor);
return !name.isUnnamed && (name.length || parent.isEmpty) ? name : translator.context.generateAnonymousName(cursor);
}
}
1 change: 1 addition & 0 deletions dstep/translator/Enum.d
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@ import dstep.translator.Declaration;
import dstep.translator.Output;
import dstep.translator.Translator;
import dstep.translator.Type;
import dstep.translator.Util;

void translateEnumConstantDecl(
Output output,
Expand Down
19 changes: 14 additions & 5 deletions dstep/translator/Record.d
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@ import dstep.translator.Declaration;
import dstep.translator.Output;
import dstep.translator.Translator;
import dstep.translator.Type;
import dstep.translator.Util;

string translateRecordTypeKeyword(in Cursor cursor)
{
Expand Down Expand Up @@ -140,7 +141,14 @@ void translateRecordDef(
auto canonical = cursor.canonical;

auto spelling = keepUnnamed ? "" : context.translateTagSpelling(cursor);
spelling = spelling == "" ? spelling : " " ~ spelling;
/* Spaghetti warning: Since unnamed values might in at least clang 19.1
* contain "unnamed enum" or something as the name instead of being empty,
* explicitly set it our variable to the empty string.
*
* Preferably, this would be disentagled and it would be ensured that there
* is only one source of the spelling in this whole function.
*/
spelling = spelling.isUnnamed ? "" : " " ~ spelling;
auto type = translateRecordTypeKeyword(cursor);

output.subscopeStrong(cursor.extent, "%s%s", type, spelling) in {
Expand Down Expand Up @@ -195,7 +203,7 @@ void translateRecordDef(

case CXCursorKind.unionDecl:
case CXCursorKind.structDecl:
if (cursor.type.isAnonymous)
if (cursor.type.isAnonymous || cursor.spelling.isUnnamed)
translateAnonymousRecord(output, context, cursor);

break;
Expand All @@ -216,7 +224,8 @@ void translateRecordDecl(Output output, Context context, Cursor cursor)
context.markAsDefined(cursor);

auto spelling = context.translateTagSpelling(cursor);
spelling = spelling == "" ? spelling : " " ~ spelling;
//spelling = spelling == "" ? spelling : " " ~ spelling;
spelling = spelling.isUnnamed ? "" : " " ~ spelling;
auto type = translateRecordTypeKeyword(cursor);
output.singleLine(cursor.extent, "%s%s;", type, spelling);
}
Expand All @@ -233,7 +242,7 @@ bool shouldSkipRecord(Context context, Cursor cursor)
cursor.kind == CXCursorKind.unionDecl)
{
auto typedefp = context.typedefParent(cursor.canonical);
auto spelling = typedefp.isValid && cursor.spelling == ""
auto spelling = typedefp.isValid && isUnnamed(cursor.spelling)
? typedefp.spelling
: cursor.spelling;

Expand All @@ -249,7 +258,7 @@ bool shouldSkipRecordDefinition(Context context, Cursor cursor)
cursor.kind == CXCursorKind.unionDecl)
{
auto typedefp = context.typedefParent(cursor.canonical);
auto spelling = typedefp.isValid && cursor.spelling == ""
auto spelling = typedefp.isValid && cursor.spelling.isUnnamed
? typedefp.spelling
: cursor.spelling;

Expand Down
21 changes: 21 additions & 0 deletions dstep/translator/Util.d
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
module dstep.translator.Util;

/** In recent versions of clang (at least 19.1.7), unnamed (anonymous) types will not
* set the spelling field to an empty string, but instead to a message like
* "enum (unnamed at [declaration_file.h:48]"
* "struct (anonymous at [declaration_file.h:48]"
* Either unnamed or anonymous is used.
* This applies to structs, enums and unions.
*
* structs and unions additionally have the isAnonymous() helper which doesn't for enums.
*
* Usage:
* cursor.spelling.isUnnamed()
*/
bool isUnnamed(string s){
import std.algorithm.searching;
return s == "" ||
s.canFind("(unnamed at") ||
s.canFind("(anonymous at");
}

2 changes: 1 addition & 1 deletion tests/unit/issues/Issue8.d
Original file line number Diff line number Diff line change
Expand Up @@ -33,7 +33,7 @@ typedef struct rd_kafka_metadata {
char *orig_broker_name; /* Name of originating broker */
} rd_kafka_metadata_t;

rd_kafka_metadata (rd_kafka_t *rk, int all_topics,
int rd_kafka_metadata (rd_kafka_t *rk, int all_topics,
rd_kafka_topic_t *only_rkt,
const struct rd_kafka_metadata **metadatap,
int timeout_ms);
Expand Down