Commit ab872f2f authored by Ken Thomases's avatar Ken Thomases Committed by Alexandre Julliard

dinput: Overhaul how the array of Mac device elements is built and sorted.

Use three separate arrays instead of one array with three sections that are separately tracked. Use the Core Foundation sort function rather than implementing a clumsy sort of our own.
parent bf938311
...@@ -108,7 +108,7 @@ struct JoystickImpl ...@@ -108,7 +108,7 @@ struct JoystickImpl
/* osx private */ /* osx private */
int id; int id;
CFMutableArrayRef elements; CFArrayRef elements;
ObjProps **propmap; ObjProps **propmap;
FFDeviceObjectReference ff; FFDeviceObjectReference ff;
struct list effects; struct list effects;
...@@ -434,35 +434,23 @@ static int get_osx_device_name(int id, char *name, int length) ...@@ -434,35 +434,23 @@ static int get_osx_device_name(int id, char *name, int length)
return 0; return 0;
} }
static void insert_sort_button(int header, IOHIDElementRef element, static CFComparisonResult button_usage_comparator(const void *val1, const void *val2, void *context)
CFMutableArrayRef elements, int index,
int target)
{ {
IOHIDElementRef targetElement; IOHIDElementRef element1 = (IOHIDElementRef)val1, element2 = (IOHIDElementRef)val2;
int usage; int usage1 = IOHIDElementGetUsage(element1), usage2 = IOHIDElementGetUsage(element2);
CFArraySetValueAtIndex(elements, header+index, NULL); if (usage1 < usage2)
targetElement = ( IOHIDElementRef ) CFArrayGetValueAtIndex( elements, header+target); return kCFCompareLessThan;
if (targetElement == NULL) if (usage1 > usage2)
{ return kCFCompareGreaterThan;
CFArraySetValueAtIndex(elements, header+target,element); return kCFCompareEqualTo;
return;
}
usage = IOHIDElementGetUsage( targetElement );
usage --; /* usage 1 based index */
insert_sort_button(header, targetElement, elements, target, usage);
CFArraySetValueAtIndex(elements, header+target,element);
} }
static void get_osx_device_elements(JoystickImpl *device, int axis_map[8]) static void get_osx_device_elements(JoystickImpl *device, int axis_map[8])
{ {
IOHIDElementRef device_main_element; IOHIDElementRef device_main_element;
CFMutableArrayRef elements; CFMutableArrayRef elements;
DWORD axes = 0;
DWORD sliders = 0; DWORD sliders = 0;
DWORD buttons = 0;
DWORD povs = 0;
device->elements = NULL; device->elements = NULL;
...@@ -480,8 +468,9 @@ static void get_osx_device_elements(JoystickImpl *device, int axis_map[8]) ...@@ -480,8 +468,9 @@ static void get_osx_device_elements(JoystickImpl *device, int axis_map[8])
if (elements) if (elements)
{ {
CFIndex idx, cnt = CFArrayGetCount( elements ); CFIndex idx, cnt = CFArrayGetCount( elements );
/* build our element array in the order that dinput expects */ CFMutableArrayRef axes = CFArrayCreateMutable(kCFAllocatorDefault, 0, &kCFTypeArrayCallBacks);
device->elements = CFArrayCreateMutable(NULL,0,NULL); CFMutableArrayRef buttons = CFArrayCreateMutable(kCFAllocatorDefault, 0, &kCFTypeArrayCallBacks);
CFMutableArrayRef povs = CFArrayCreateMutable(kCFAllocatorDefault, 0, &kCFTypeArrayCallBacks);
for ( idx = 0; idx < cnt; idx++ ) for ( idx = 0; idx < cnt; idx++ )
{ {
...@@ -498,17 +487,13 @@ static void get_osx_device_elements(JoystickImpl *device, int axis_map[8]) ...@@ -498,17 +487,13 @@ static void get_osx_device_elements(JoystickImpl *device, int axis_map[8])
continue; continue;
} }
if (buttons < 128) if (CFArrayGetCount(buttons) < 128)
{ CFArrayAppendValue(buttons, element);
CFArrayInsertValueAtIndex(device->elements, (axes+povs+buttons), element);
buttons++;
}
break; break;
} }
case kIOHIDElementTypeInput_Axis: case kIOHIDElementTypeInput_Axis:
{ {
CFArrayInsertValueAtIndex(device->elements, axes, element); CFArrayAppendValue(axes, element);
axes++;
break; break;
} }
case kIOHIDElementTypeInput_Misc: case kIOHIDElementTypeInput_Misc:
...@@ -518,8 +503,7 @@ static void get_osx_device_elements(JoystickImpl *device, int axis_map[8]) ...@@ -518,8 +503,7 @@ static void get_osx_device_elements(JoystickImpl *device, int axis_map[8])
{ {
case kHIDUsage_GD_Hatswitch: case kHIDUsage_GD_Hatswitch:
{ {
CFArrayInsertValueAtIndex(device->elements, (axes+povs), element); CFArrayAppendValue(povs, element);
povs++;
break; break;
} }
case kHIDUsage_GD_Slider: case kHIDUsage_GD_Slider:
...@@ -534,9 +518,8 @@ static void get_osx_device_elements(JoystickImpl *device, int axis_map[8]) ...@@ -534,9 +518,8 @@ static void get_osx_device_elements(JoystickImpl *device, int axis_map[8])
case kHIDUsage_GD_Ry: case kHIDUsage_GD_Ry:
case kHIDUsage_GD_Rz: case kHIDUsage_GD_Rz:
{ {
CFArrayInsertValueAtIndex(device->elements, axes, element); axis_map[CFArrayGetCount(axes)]=usage;
axis_map[axes]=usage; CFArrayAppendValue(axes, element);
axes++;
break; break;
} }
default: default:
...@@ -548,22 +531,29 @@ static void get_osx_device_elements(JoystickImpl *device, int axis_map[8]) ...@@ -548,22 +531,29 @@ static void get_osx_device_elements(JoystickImpl *device, int axis_map[8])
FIXME("Unhandled type %i\n",type); FIXME("Unhandled type %i\n",type);
} }
} }
}
device->generic.devcaps.dwAxes = axes;
device->generic.devcaps.dwButtons = buttons;
device->generic.devcaps.dwPOVs = povs;
/* Sort buttons into correct order */ /* Sort buttons into correct order */
for (buttons = 0; buttons < device->generic.devcaps.dwButtons; buttons++) CFArraySortValues(buttons, CFRangeMake(0, CFArrayGetCount(buttons)), button_usage_comparator, NULL);
{
IOHIDElementRef element = ( IOHIDElementRef ) CFArrayGetValueAtIndex( device->elements, axes+povs+buttons); device->generic.devcaps.dwAxes = CFArrayGetCount(axes);
uint32_t usage = IOHIDElementGetUsage( element ); device->generic.devcaps.dwButtons = CFArrayGetCount(buttons);
usage --; /* usage is 1 indexed we need 0 indexed */ device->generic.devcaps.dwPOVs = CFArrayGetCount(povs);
if (usage == buttons)
continue;
insert_sort_button(axes+povs, element, device->elements,buttons,usage); /* build our element array in the order that dinput expects */
CFArrayAppendArray(axes, povs, CFRangeMake(0, device->generic.devcaps.dwPOVs));
CFArrayAppendArray(axes, buttons, CFRangeMake(0, device->generic.devcaps.dwButtons));
device->elements = axes;
axes = NULL;
CFRelease(povs);
CFRelease(buttons);
CFRelease(elements);
}
else
{
device->generic.devcaps.dwAxes = 0;
device->generic.devcaps.dwButtons = 0;
device->generic.devcaps.dwPOVs = 0;
} }
} }
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment