Skip to content
Projects
Groups
Snippets
Help
This project
Loading...
Sign in / Register
Toggle navigation
W
wine-winehq
Project
Project
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Registry
Registry
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
wine
wine-winehq
Commits
9c77c445
Commit
9c77c445
authored
Jun 21, 2021
by
Rémi Bernon
Committed by
Alexandre Julliard
Jun 23, 2021
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
hidclass.sys: Support parsing of explicit usage page.
Signed-off-by:
Rémi Bernon
<
rbernon@codeweavers.com
>
Signed-off-by:
Alexandre Julliard
<
julliard@winehq.org
>
parent
2731a014
Hide whitespace changes
Inline
Side-by-side
Showing
1 changed file
with
43 additions
and
10 deletions
+43
-10
descriptor.c
dlls/hidclass.sys/descriptor.c
+43
-10
No files found.
dlls/hidclass.sys/descriptor.c
View file @
9c77c445
...
@@ -272,7 +272,9 @@ struct hid_parser_state
...
@@ -272,7 +272,9 @@ struct hid_parser_state
{
{
HIDP_CAPS
caps
;
HIDP_CAPS
caps
;
USAGE
usages
[
256
];
USAGE
usages_page
[
256
];
USAGE
usages_min
[
256
];
USAGE
usages_max
[
256
];
DWORD
usages_size
;
DWORD
usages_size
;
struct
hid_value_caps
items
;
struct
hid_value_caps
items
;
...
@@ -323,6 +325,9 @@ static void reset_local_items( struct hid_parser_state *state )
...
@@ -323,6 +325,9 @@ static void reset_local_items( struct hid_parser_state *state )
memset
(
&
state
->
items
,
0
,
sizeof
(
state
->
items
)
);
memset
(
&
state
->
items
,
0
,
sizeof
(
state
->
items
)
);
copy_global_items
(
&
state
->
items
,
&
tmp
);
copy_global_items
(
&
state
->
items
,
&
tmp
);
copy_collection_items
(
&
state
->
items
,
&
tmp
);
copy_collection_items
(
&
state
->
items
,
&
tmp
);
memset
(
&
state
->
usages_page
,
0
,
sizeof
(
state
->
usages_page
)
);
memset
(
&
state
->
usages_min
,
0
,
sizeof
(
state
->
usages_min
)
);
memset
(
&
state
->
usages_max
,
0
,
sizeof
(
state
->
usages_max
)
);
state
->
usages_size
=
0
;
state
->
usages_size
=
0
;
}
}
...
@@ -352,14 +357,42 @@ static BOOL parse_global_pop( struct hid_parser_state *state )
...
@@ -352,14 +357,42 @@ static BOOL parse_global_pop( struct hid_parser_state *state )
return
TRUE
;
return
TRUE
;
}
}
static
BOOL
parse_local_usage
(
struct
hid_parser_state
*
state
,
USAGE
usage
)
static
BOOL
parse_local_usage
(
struct
hid_parser_state
*
state
,
USAGE
usage
_page
,
USAGE
usage
)
{
{
state
->
usages
[
state
->
usages_size
]
=
usage
;
if
(
!
usage_page
)
usage_page
=
state
->
items
.
usage_page
;
if
(
state
->
items
.
is_range
)
state
->
usages_size
=
0
;
state
->
usages_page
[
state
->
usages_size
]
=
usage_page
;
state
->
usages_min
[
state
->
usages_size
]
=
usage
;
state
->
usages_max
[
state
->
usages_size
]
=
usage
;
state
->
items
.
usage_min
=
usage
;
state
->
items
.
usage_max
=
usage
;
state
->
items
.
is_range
=
FALSE
;
state
->
items
.
is_range
=
FALSE
;
if
(
state
->
usages_size
++
==
255
)
ERR
(
"HID parser usages stack overflow!
\n
"
);
if
(
state
->
usages_size
++
==
255
)
ERR
(
"HID parser usages stack overflow!
\n
"
);
return
state
->
usages_size
<=
255
;
return
state
->
usages_size
<=
255
;
}
}
static
void
parse_local_usage_min
(
struct
hid_parser_state
*
state
,
USAGE
usage_page
,
USAGE
usage
)
{
if
(
!
usage_page
)
usage_page
=
state
->
items
.
usage_page
;
if
(
!
state
->
items
.
is_range
)
state
->
usages_max
[
0
]
=
0
;
state
->
usages_page
[
0
]
=
usage_page
;
state
->
usages_min
[
0
]
=
usage
;
state
->
items
.
usage_min
=
usage
;
state
->
items
.
is_range
=
TRUE
;
state
->
usages_size
=
1
;
}
static
void
parse_local_usage_max
(
struct
hid_parser_state
*
state
,
USAGE
usage_page
,
USAGE
usage
)
{
if
(
!
usage_page
)
usage_page
=
state
->
items
.
usage_page
;
if
(
!
state
->
items
.
is_range
)
state
->
usages_min
[
0
]
=
0
;
state
->
usages_page
[
0
]
=
usage_page
;
state
->
usages_max
[
0
]
=
usage
;
state
->
items
.
usage_max
=
usage
;
state
->
items
.
is_range
=
TRUE
;
state
->
usages_size
=
1
;
}
static
BOOL
parse_new_collection
(
struct
hid_parser_state
*
state
)
static
BOOL
parse_new_collection
(
struct
hid_parser_state
*
state
)
{
{
if
(
!
array_reserve
(
&
state
->
stack
,
&
state
->
stack_size
,
state
->
collection_idx
))
if
(
!
array_reserve
(
&
state
->
stack
,
&
state
->
stack_size
,
state
->
collection_idx
))
...
@@ -377,6 +410,9 @@ static BOOL parse_new_collection( struct hid_parser_state *state )
...
@@ -377,6 +410,9 @@ static BOOL parse_new_collection( struct hid_parser_state *state )
copy_collection_items
(
state
->
stack
+
state
->
collection_idx
,
&
state
->
items
);
copy_collection_items
(
state
->
stack
+
state
->
collection_idx
,
&
state
->
items
);
state
->
collection_idx
++
;
state
->
collection_idx
++
;
state
->
items
.
usage_min
=
state
->
usages_min
[
0
];
state
->
items
.
usage_max
=
state
->
usages_max
[
0
];
state
->
collections
[
state
->
caps
.
NumberLinkCollectionNodes
]
=
state
->
items
;
state
->
collections
[
state
->
caps
.
NumberLinkCollectionNodes
]
=
state
->
items
;
state
->
items
.
link_collection
=
state
->
caps
.
NumberLinkCollectionNodes
;
state
->
items
.
link_collection
=
state
->
caps
.
NumberLinkCollectionNodes
;
state
->
items
.
link_usage_page
=
state
->
items
.
usage_page
;
state
->
items
.
link_usage_page
=
state
->
items
.
usage_page
;
...
@@ -417,8 +453,8 @@ static BOOL parse_new_value_caps( struct hid_parser_state *state, HIDP_REPORT_TY
...
@@ -417,8 +453,8 @@ static BOOL parse_new_value_caps( struct hid_parser_state *state, HIDP_REPORT_TY
list_add_tail
(
&
collection
->
features
,
&
feature
->
entry
);
list_add_tail
(
&
collection
->
features
,
&
feature
->
entry
);
feature
->
type
=
type
;
feature
->
type
=
type
;
feature
->
isData
=
((
state
->
items
.
bit_field
&
INPUT_DATA_CONST
)
==
0
);
feature
->
isData
=
((
state
->
items
.
bit_field
&
INPUT_DATA_CONST
)
==
0
);
if
(
j
<
state
->
usages_size
)
state
->
items
.
usage_min
=
state
->
usages
[
j
];
copy_hidp_value_caps
(
&
feature
->
caps
,
&
state
->
items
);
copy_hidp_value_caps
(
&
feature
->
caps
,
&
state
->
items
);
if
(
j
<
state
->
usages_size
)
feature
->
caps
.
NotRange
.
Usage
=
state
->
usages_min
[
j
];
feature
->
caps
.
ReportCount
=
1
;
feature
->
caps
.
ReportCount
=
1
;
if
(
j
+
1
>=
state
->
usages_size
)
if
(
j
+
1
>=
state
->
usages_size
)
{
{
...
@@ -508,7 +544,6 @@ static int parse_descriptor( BYTE *descriptor, unsigned int index, unsigned int
...
@@ -508,7 +544,6 @@ static int parse_descriptor( BYTE *descriptor, unsigned int index, unsigned int
subcollection
->
parent
=
collection
;
subcollection
->
parent
=
collection
;
/* Only set our collection once...
/* Only set our collection once...
We do not properly handle composite devices yet. */
We do not properly handle composite devices yet. */
if
(
state
->
usages_size
)
state
->
items
.
usage_min
=
state
->
usages
[
state
->
usages_size
-
1
];
list_init
(
&
subcollection
->
features
);
list_init
(
&
subcollection
->
features
);
list_init
(
&
subcollection
->
collections
);
list_init
(
&
subcollection
->
collections
);
parse_collection
(
size
,
value
,
subcollection
);
parse_collection
(
size
,
value
,
subcollection
);
...
@@ -559,15 +594,13 @@ static int parse_descriptor( BYTE *descriptor, unsigned int index, unsigned int
...
@@ -559,15 +594,13 @@ static int parse_descriptor( BYTE *descriptor, unsigned int index, unsigned int
break
;
break
;
case
SHORT_ITEM
(
TAG_LOCAL_USAGE
,
TAG_TYPE_LOCAL
):
case
SHORT_ITEM
(
TAG_LOCAL_USAGE
,
TAG_TYPE_LOCAL
):
if
(
!
parse_local_usage
(
state
,
value
))
return
-
1
;
if
(
!
parse_local_usage
(
state
,
value
>>
16
,
value
&
0xffff
))
return
-
1
;
break
;
break
;
case
SHORT_ITEM
(
TAG_LOCAL_USAGE_MINIMUM
,
TAG_TYPE_LOCAL
):
case
SHORT_ITEM
(
TAG_LOCAL_USAGE_MINIMUM
,
TAG_TYPE_LOCAL
):
state
->
items
.
usage_min
=
value
;
parse_local_usage_min
(
state
,
value
>>
16
,
value
&
0xffff
);
state
->
items
.
is_range
=
TRUE
;
break
;
break
;
case
SHORT_ITEM
(
TAG_LOCAL_USAGE_MAXIMUM
,
TAG_TYPE_LOCAL
):
case
SHORT_ITEM
(
TAG_LOCAL_USAGE_MAXIMUM
,
TAG_TYPE_LOCAL
):
state
->
items
.
usage_max
=
value
;
parse_local_usage_max
(
state
,
value
>>
16
,
value
&
0xffff
);
state
->
items
.
is_range
=
TRUE
;
break
;
break
;
case
SHORT_ITEM
(
TAG_LOCAL_DESIGNATOR_INDEX
,
TAG_TYPE_LOCAL
):
case
SHORT_ITEM
(
TAG_LOCAL_DESIGNATOR_INDEX
,
TAG_TYPE_LOCAL
):
state
->
items
.
designator_min
=
state
->
items
.
designator_max
=
value
;
state
->
items
.
designator_min
=
state
->
items
.
designator_max
=
value
;
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment