Browse Source

[usb] Select preferred USB device configuration based on driver score

Generate a score for each possible USB device configuration based on
the available driver support, and select the configuration with the
highest score.  This will allow us to prefer ECM over RNDIS (for
devices which support both) and will allow us to meaningfully select a
configuration even when we have drivers available for all functions
(e.g. when exposing unused functions via EFI_USB_IO_PROTOCOL).

Signed-off-by: Michael Brown <mcb30@ipxe.org>
tags/v1.20.1
Michael Brown 9 years ago
parent
commit
549a0caabb

+ 254
- 146
src/drivers/bus/usb.c View File

@@ -900,76 +900,155 @@ int usb_get_string_descriptor ( struct usb_device *usb, unsigned int index,
900 900
  ******************************************************************************
901 901
  */
902 902
 
903
+/**
904
+ * Get USB configuration descriptor
905
+ *
906
+ * @v usb		USB device
907
+ * @v index		Configuration index
908
+ * @ret config		Configuration descriptor
909
+ * @ret rc		Return status code
910
+ *
911
+ * The configuration descriptor is dynamically allocated and must
912
+ * eventually be freed by the caller.
913
+ */
914
+static int
915
+usb_config_descriptor ( struct usb_device *usb, unsigned int index,
916
+			struct usb_configuration_descriptor **config ) {
917
+	struct usb_configuration_descriptor partial;
918
+	size_t len;
919
+	int rc;
920
+
921
+	/* Read first part of configuration descriptor to get size */
922
+	if ( ( rc = usb_get_config_descriptor ( usb, index, &partial,
923
+						sizeof ( partial ) ) ) != 0 ) {
924
+		DBGC ( usb, "USB %s could not get configuration descriptor %d: "
925
+		       "%s\n", usb->name, index, strerror ( rc ) );
926
+		goto err_get_partial;
927
+	}
928
+	len = le16_to_cpu ( partial.len );
929
+	if ( len < sizeof ( partial ) ) {
930
+		DBGC ( usb, "USB %s underlength configuraton descriptor %d\n",
931
+		       usb->name, index );
932
+		rc = -EINVAL;
933
+		goto err_partial_len;
934
+	}
935
+
936
+	/* Allocate buffer for whole configuration descriptor */
937
+	*config = malloc ( len );
938
+	if ( ! *config ) {
939
+		rc = -ENOMEM;
940
+		goto err_alloc_config;
941
+	}
942
+
943
+	/* Read whole configuration descriptor */
944
+	if ( ( rc = usb_get_config_descriptor ( usb, index, *config,
945
+						len ) ) != 0 ) {
946
+		DBGC ( usb, "USB %s could not get configuration descriptor %d: "
947
+		       "%s\n", usb->name, index, strerror ( rc ) );
948
+		goto err_get_config_descriptor;
949
+	}
950
+	if ( (*config)->len != partial.len ) {
951
+		DBGC ( usb, "USB %s bad configuration descriptor %d length\n",
952
+		       usb->name, index );
953
+		rc = -EINVAL;
954
+		goto err_config_len;
955
+	}
956
+
957
+	return 0;
958
+
959
+ err_config_len:
960
+ err_get_config_descriptor:
961
+	free ( *config );
962
+ err_alloc_config:
963
+ err_partial_len:
964
+ err_get_partial:
965
+	return rc;
966
+}
967
+
903 968
 /**
904 969
  * Describe USB function
905 970
  *
906
- * @v func		USB function
971
+ * @v usb		USB device
907 972
  * @v config		Configuration descriptor
908 973
  * @v first		First interface number
974
+ * @v interfaces	Interface list to fill in
975
+ * @v desc		Function descriptor to fill in
909 976
  * @ret rc		Return status code
910 977
  */
911
-static int usb_function ( struct usb_function *func,
978
+static int usb_describe ( struct usb_device *usb,
912 979
 			  struct usb_configuration_descriptor *config,
913
-			  unsigned int first ) {
914
-	struct usb_device *usb = func->usb;
980
+			  unsigned int first, uint8_t *interfaces,
981
+			  struct usb_function_descriptor *desc ) {
915 982
 	struct usb_interface_association_descriptor *association;
916 983
 	struct usb_interface_descriptor *interface;
917 984
 	struct cdc_union_descriptor *cdc_union;
918 985
 	unsigned int i;
919 986
 
987
+	/* Fill in vendor and product ID */
988
+	desc->vendor = le16_to_cpu ( usb->device.vendor );
989
+	desc->product = le16_to_cpu ( usb->device.product );
990
+
920 991
 	/* First, look for an interface association descriptor */
921 992
 	association = usb_interface_association_descriptor ( config, first );
922 993
 	if ( association ) {
923 994
 
924 995
 		/* Sanity check */
925
-		if ( association->count > config->interfaces ) {
996
+		assert ( association->first == first );
997
+		if ( ( first + association->count ) > config->interfaces ) {
926 998
 			DBGC ( usb, "USB %s has invalid association [%d-%d)\n",
927
-			       func->name, association->first,
928
-			       ( association->first + association->count ) );
999
+			       usb->name, first, ( first + association->count));
929 1000
 			return -ERANGE;
930 1001
 		}
931 1002
 
932 1003
 		/* Describe function */
933
-		memcpy ( &func->class, &association->class,
934
-			 sizeof ( func->class ) );
935
-		func->count = association->count;
1004
+		memcpy ( &desc->class, &association->class,
1005
+			 sizeof ( desc->class ) );
1006
+		desc->count = association->count;
936 1007
 		for ( i = 0 ; i < association->count ; i++ )
937
-			func->interface[i] = ( association->first + i );
1008
+			interfaces[i] = ( first + i );
938 1009
 		return 0;
939 1010
 	}
940 1011
 
941 1012
 	/* Next, look for an interface descriptor */
942 1013
 	interface = usb_interface_descriptor ( config, first, 0 );
943 1014
 	if ( ! interface ) {
944
-		DBGC ( usb, "USB %s has no interface descriptor\n",
945
-		       func->name );
1015
+		DBGC ( usb, "USB %s has no descriptor for interface %d\n",
1016
+		       usb->name, first );
946 1017
 		return -ENOENT;
947 1018
 	}
948 1019
 
949 1020
 	/* Describe function */
950
-	memcpy ( &func->class, &interface->class, sizeof ( func->class ) );
951
-	func->count = 1;
952
-	func->interface[0] = first;
1021
+	memcpy ( &desc->class, &interface->class, sizeof ( desc->class ) );
1022
+	desc->count = 1;
1023
+	interfaces[0] = first;
953 1024
 
954 1025
 	/* Look for a CDC union descriptor, if applicable */
955
-	if ( ( func->class.class == USB_CLASS_CDC ) &&
1026
+	if ( ( desc->class.class == USB_CLASS_CDC ) &&
956 1027
 	     ( cdc_union = cdc_union_descriptor ( config, interface ) ) ) {
957 1028
 
958 1029
 		/* Determine interface count */
959
-		func->count = ( ( cdc_union->header.len -
1030
+		desc->count = ( ( cdc_union->header.len -
960 1031
 				  offsetof ( typeof ( *cdc_union ),
961 1032
 					     interface[0] ) ) /
962 1033
 				sizeof ( cdc_union->interface[0] ) );
963
-		if ( func->count > config->interfaces ) {
1034
+		if ( desc->count > config->interfaces ) {
964 1035
 			DBGC ( usb, "USB %s has invalid union functional "
965 1036
 			       "descriptor with %d interfaces\n",
966
-			       func->name, func->count );
1037
+			       usb->name, desc->count );
967 1038
 			return -ERANGE;
968 1039
 		}
969 1040
 
970 1041
 		/* Describe function */
971
-		for ( i = 0 ; i < func->count ; i++ )
972
-			func->interface[i] = cdc_union->interface[i];
1042
+		for ( i = 0 ; i < desc->count ; i++ ) {
1043
+			if ( cdc_union->interface[i] >= config->interfaces ) {
1044
+				DBGC ( usb, "USB %s has invalid union "
1045
+				       "functional descriptor covering "
1046
+				       "interface %d\n", usb->name,
1047
+				       cdc_union->interface[i] );
1048
+				return -ERANGE;
1049
+			}
1050
+			interfaces[i] = cdc_union->interface[i];
1051
+		}
973 1052
 
974 1053
 		return 0;
975 1054
 	}
@@ -977,17 +1056,38 @@ static int usb_function ( struct usb_function *func,
977 1056
 	return 0;
978 1057
 }
979 1058
 
1059
+/**
1060
+ * Update list of used interface
1061
+ *
1062
+ * @v usb		USB device
1063
+ * @v count		Number of interfaces
1064
+ * @v interface		List of interfaces
1065
+ * @v used		List of already-used interfaces
1066
+ * @ret rc		Return status code
1067
+ */
1068
+static int usb_used ( struct usb_device *usb, unsigned int count,
1069
+		      uint8_t *interface, uint8_t *used ) {
1070
+	unsigned int i;
1071
+
1072
+	for ( i = 0 ; i < count ; i++ ) {
1073
+		if ( used[interface[i]] ) {
1074
+			DBGC ( usb, "USB %s interface %d already in use\n",
1075
+			       usb->name, interface[i] );
1076
+			return -EINVAL;
1077
+		}
1078
+		used[interface[i]] = 1;
1079
+	}
1080
+	return 0;
1081
+}
1082
+
980 1083
 /**
981 1084
  * Find USB device driver
982 1085
  *
983
- * @v vendor		Vendor ID
984
- * @v product		Product ID
985
- * @v class		Class
1086
+ * @v desc		Function descriptor
986 1087
  * @ret id		USB device ID, or NULL
987 1088
  * @ret driver		USB device driver, or NULL
988 1089
  */
989
-struct usb_driver * usb_find_driver ( unsigned int vendor, unsigned int product,
990
-				      struct usb_class *class,
1090
+struct usb_driver * usb_find_driver ( struct usb_function_descriptor *desc,
991 1091
 				      struct usb_device_id **id ) {
992 1092
 	struct usb_driver *driver;
993 1093
 	unsigned int i;
@@ -998,13 +1098,13 @@ struct usb_driver * usb_find_driver ( unsigned int vendor, unsigned int product,
998 1098
 
999 1099
 			/* Check for a matching ID */
1000 1100
 			*id = &driver->ids[i];
1001
-			if ( ( ( (*id)->vendor == vendor ) ||
1101
+			if ( ( ( (*id)->vendor == desc->vendor ) ||
1002 1102
 			       ( (*id)->vendor == USB_ANY_ID ) ) &&
1003
-			     ( ( (*id)->product == product ) ||
1103
+			     ( ( (*id)->product == desc->product ) ||
1004 1104
 			       ( (*id)->product == USB_ANY_ID ) ) &&
1005
-			     ( (*id)->class.class == class->class ) &&
1006
-			     ( (*id)->class.subclass == class->subclass ) &&
1007
-			     ( (*id)->class.protocol == class->protocol ) )
1105
+			     ( (*id)->class.class == desc->class.class ) &&
1106
+			     ( (*id)->class.subclass == desc->class.subclass )&&
1107
+			     ( (*id)->class.protocol == desc->class.protocol ) )
1008 1108
 				return driver;
1009 1109
 		}
1010 1110
 	}
@@ -1014,6 +1114,51 @@ struct usb_driver * usb_find_driver ( unsigned int vendor, unsigned int product,
1014 1114
 	return NULL;
1015 1115
 }
1016 1116
 
1117
+/**
1118
+ * Get USB device configuration score
1119
+ *
1120
+ * @v usb		USB device
1121
+ * @v config		Configuration descriptor
1122
+ * @ret score		Device configuration score, or negative error
1123
+ */
1124
+static int usb_score ( struct usb_device *usb,
1125
+		       struct usb_configuration_descriptor *config ) {
1126
+	uint8_t used[config->interfaces];
1127
+	uint8_t interface[config->interfaces];
1128
+	struct usb_function_descriptor desc;
1129
+	struct usb_driver *driver;
1130
+	struct usb_device_id *id;
1131
+	unsigned int first;
1132
+	unsigned int score = 0;
1133
+	int rc;
1134
+
1135
+	/* Identify each function in turn */
1136
+	memset ( used, 0, sizeof ( used ) );
1137
+	for ( first = 0 ; first < config->interfaces ; first++ ) {
1138
+
1139
+		/* Skip interfaces already used */
1140
+		if ( used[first] )
1141
+			continue;
1142
+
1143
+		/* Describe function */
1144
+		if ( ( rc = usb_describe ( usb, config, first, interface,
1145
+					   &desc ) ) != 0 )
1146
+			return rc;
1147
+
1148
+		/* Update used interfaces */
1149
+		if ( ( rc = usb_used ( usb, desc.count, interface,
1150
+				       used ) ) != 0 )
1151
+			return rc;
1152
+
1153
+		/* Look for a driver for this function */
1154
+		driver = usb_find_driver ( &desc, &id );
1155
+		if ( driver )
1156
+			score += driver->score;
1157
+	}
1158
+
1159
+	return score;
1160
+}
1161
+
1017 1162
 /**
1018 1163
  * Probe USB device driver
1019 1164
  *
@@ -1029,13 +1174,12 @@ static int usb_probe ( struct usb_function *func,
1029 1174
 	int rc;
1030 1175
 
1031 1176
 	/* Identify driver */
1032
-	driver = usb_find_driver ( func->dev.desc.vendor, func->dev.desc.device,
1033
-				   &func->class, &id );
1177
+	driver = usb_find_driver ( &func->desc, &id );
1034 1178
 	if ( ! driver ) {
1035 1179
 		DBGC ( usb, "USB %s %04x:%04x class %d:%d:%d has no driver\n",
1036
-		       func->name, func->dev.desc.vendor, func->dev.desc.device,
1037
-		       func->class.class, func->class.subclass,
1038
-		       func->class.protocol );
1180
+		       func->name, func->desc.vendor, func->desc.product,
1181
+		       func->desc.class.class, func->desc.class.subclass,
1182
+		       func->desc.class.protocol );
1039 1183
 		return -ENOENT;
1040 1184
 	}
1041 1185
 
@@ -1106,28 +1250,24 @@ usb_probe_all ( struct usb_device *usb,
1106 1250
 		list_add_tail ( &func->list, &usb->functions );
1107 1251
 
1108 1252
 		/* Identify function */
1109
-		if ( ( rc = usb_function ( func, config, first ) ) != 0 )
1110
-			goto err_function;
1111
-		assert ( func->count <= config->interfaces );
1253
+		if ( ( rc = usb_describe ( usb, config, first, func->interface,
1254
+					   &func->desc ) ) != 0 )
1255
+			goto err_describe;
1256
+		assert ( func->desc.count <= config->interfaces );
1112 1257
 
1113 1258
 		/* Mark interfaces as used */
1114
-		for ( i = 0 ; i < func->count ; i++ ) {
1115
-			if ( func->interface[i] >= config->interfaces ) {
1116
-				DBGC ( usb, "USB %s has invalid interface %d\n",
1117
-				       func->name, func->interface[i] );
1118
-				goto err_interface;
1119
-			}
1120
-			used[ func->interface[i] ] = 1;
1121
-		}
1259
+		if ( ( rc = usb_used ( usb, func->desc.count, func->interface,
1260
+				       used ) ) != 0 )
1261
+			goto err_used;
1122 1262
 
1123 1263
 		/* Probe device driver */
1124 1264
 		if ( ( rc = usb_probe ( func, config ) ) != 0 )
1125 1265
 			goto err_probe;
1126 1266
 		DBGC ( usb, "USB %s %04x:%04x class %d:%d:%d interfaces ",
1127
-		       func->name, func->dev.desc.vendor, func->dev.desc.device,
1128
-		       func->class.class, func->class.subclass,
1129
-		       func->class.protocol );
1130
-		for ( i = 0 ; i < func->count ; i++ )
1267
+		       func->name, func->desc.vendor, func->desc.product,
1268
+		       func->desc.class.class, func->desc.class.subclass,
1269
+		       func->desc.class.protocol );
1270
+		for ( i = 0 ; i < func->desc.count ; i++ )
1131 1271
 			DBGC ( usb, "%s%d", ( i ? "," : "" ),
1132 1272
 			       func->interface[i] );
1133 1273
 		DBGC ( usb, " using driver %s\n", func->dev.driver_name );
@@ -1140,8 +1280,8 @@ usb_probe_all ( struct usb_device *usb,
1140 1280
 		list_del ( &func->dev.siblings );
1141 1281
 		usb_remove ( func );
1142 1282
 	err_probe:
1143
-	err_interface:
1144
-	err_function:
1283
+	err_used:
1284
+	err_describe:
1145 1285
 		list_del ( &func->list );
1146 1286
 		free ( func );
1147 1287
 	err_alloc:
@@ -1177,82 +1317,6 @@ static void usb_remove_all ( struct usb_device *usb ) {
1177 1317
 	}
1178 1318
 }
1179 1319
 
1180
-/**
1181
- * Select USB device configuration
1182
- *
1183
- * @v usb		USB device
1184
- * @v index		Configuration index
1185
- * @ret rc		Return status code
1186
- */
1187
-static int usb_configure ( struct usb_device *usb, unsigned int index ) {
1188
-	struct usb_configuration_descriptor partial;
1189
-	struct usb_configuration_descriptor *config;
1190
-	size_t len;
1191
-	int rc;
1192
-
1193
-	/* Read first part of configuration descriptor to get size */
1194
-	if ( ( rc = usb_get_config_descriptor ( usb, index, &partial,
1195
-						sizeof ( partial ) ) ) != 0 ) {
1196
-		DBGC ( usb, "USB %s could not get configuration descriptor %d: "
1197
-		       "%s\n", usb->name, index, strerror ( rc ) );
1198
-		goto err_get_partial;
1199
-	}
1200
-	len = le16_to_cpu ( partial.len );
1201
-	if ( len < sizeof ( partial ) ) {
1202
-		DBGC ( usb, "USB %s underlength configuraton descriptor %d\n",
1203
-		       usb->name, index );
1204
-		rc = -EINVAL;
1205
-		goto err_partial_len;
1206
-	}
1207
-
1208
-	/* Allocate buffer for whole configuration descriptor */
1209
-	config = malloc ( len );
1210
-	if ( ! config ) {
1211
-		rc = -ENOMEM;
1212
-		goto err_alloc_config;
1213
-	}
1214
-
1215
-	/* Read whole configuration descriptor */
1216
-	if ( ( rc = usb_get_config_descriptor ( usb, index, config,
1217
-						len ) ) != 0 ) {
1218
-		DBGC ( usb, "USB %s could not get configuration descriptor %d: "
1219
-		       "%s\n", usb->name, index, strerror ( rc ) );
1220
-		goto err_get_config_descriptor;
1221
-	}
1222
-	if ( config->len != partial.len ) {
1223
-		DBGC ( usb, "USB %s bad configuration descriptor %d length\n",
1224
-		       usb->name, index );
1225
-		rc = -EINVAL;
1226
-		goto err_config_len;
1227
-	}
1228
-
1229
-	/* Set configuration */
1230
-	if ( ( rc = usb_set_configuration ( usb, config->config ) ) != 0){
1231
-		DBGC ( usb, "USB %s could not set configuration %d: %s\n",
1232
-		       usb->name, config->config, strerror ( rc ) );
1233
-		goto err_set_configuration;
1234
-	}
1235
-
1236
-	/* Probe USB device drivers */
1237
-	usb_probe_all ( usb, config );
1238
-
1239
-	/* Free configuration descriptor */
1240
-	free ( config );
1241
-
1242
-	return 0;
1243
-
1244
-	usb_remove_all ( usb );
1245
-	usb_set_configuration ( usb, 0 );
1246
- err_set_configuration:
1247
- err_config_len:
1248
- err_get_config_descriptor:
1249
-	free ( config );
1250
- err_alloc_config:
1251
- err_partial_len:
1252
- err_get_partial:
1253
-	return rc;
1254
-}
1255
-
1256 1320
 /**
1257 1321
  * Clear USB device configuration
1258 1322
  *
@@ -1275,32 +1339,76 @@ static void usb_deconfigure ( struct usb_device *usb ) {
1275 1339
 }
1276 1340
 
1277 1341
 /**
1278
- * Find and select a supported USB device configuration
1342
+ * Choose our preferred USB device configuration
1279 1343
  *
1280 1344
  * @v usb		USB device
1281 1345
  * @ret rc		Return status code
1282 1346
  */
1283
-static int usb_configure_any ( struct usb_device *usb ) {
1347
+static int usb_autoconfigure ( struct usb_device *usb ) {
1348
+	struct usb_configuration_descriptor *config;
1349
+	unsigned int preferred = 0;
1284 1350
 	unsigned int index;
1285
-	int rc = -ENOENT;
1351
+	int score;
1352
+	int best = 0;
1353
+	int rc;
1286 1354
 
1287
-	/* Attempt all configuration indexes */
1355
+	/* Calculate driver score for each configuration index */
1288 1356
 	for ( index = 0 ; index < usb->device.configurations ; index++ ) {
1289 1357
 
1290
-		/* Attempt this configuration index */
1291
-		if ( ( rc = usb_configure ( usb, index ) ) != 0 )
1292
-			continue;
1358
+		/* Read configuration descriptor */
1359
+		if ( ( rc = usb_config_descriptor ( usb, index,
1360
+						    &config ) ) != 0 )
1361
+			goto err_config;
1293 1362
 
1294
-		/* If we have no drivers, then try the next configuration */
1295
-		if ( list_empty ( &usb->functions ) ) {
1296
-			rc = -ENOTSUP;
1297
-			usb_deconfigure ( usb );
1298
-			continue;
1363
+		/* Get score for this configuration */
1364
+		score = usb_score ( usb, config );
1365
+		if ( score < 0 ) {
1366
+			rc = score;
1367
+			goto err_score;
1299 1368
 		}
1369
+		DBGC2 ( usb, "USB %s configuration %d score %d\n",
1370
+			usb->name, config->config, score );
1300 1371
 
1301
-		return 0;
1372
+		/* Record as preferred configuration, if applicable */
1373
+		if ( score > best ) {
1374
+			best = score;
1375
+			preferred = index;
1376
+		}
1377
+
1378
+		/* Free configuration descriptor */
1379
+		free ( config );
1380
+		config = NULL;
1302 1381
 	}
1303 1382
 
1383
+	/* Read preferred configuration descriptor */
1384
+	if ( ( rc = usb_config_descriptor ( usb, preferred, &config ) ) != 0 )
1385
+		goto err_preferred;
1386
+
1387
+	/* Set configuration */
1388
+	if ( ( rc = usb_set_configuration ( usb, config->config ) ) != 0){
1389
+		DBGC ( usb, "USB %s could not set configuration %d: %s\n",
1390
+		       usb->name, config->config, strerror ( rc ) );
1391
+		goto err_set_configuration;
1392
+	}
1393
+
1394
+	/* Probe USB device drivers */
1395
+	usb_probe_all ( usb, config );
1396
+
1397
+	/* Free configuration descriptor */
1398
+	free ( config );
1399
+
1400
+	return 0;
1401
+
1402
+	usb_remove_all ( usb );
1403
+	usb_set_configuration ( usb, 0 );
1404
+ err_set_configuration:
1405
+	free ( config );
1406
+ err_preferred:
1407
+	return rc;
1408
+
1409
+ err_score:
1410
+	free ( config );
1411
+ err_config:
1304 1412
 	return rc;
1305 1413
 }
1306 1414
 
@@ -1444,13 +1552,13 @@ static int register_usb ( struct usb_device *usb ) {
1444 1552
 	       usb_speed_name ( port->speed ), usb->control.mtu );
1445 1553
 
1446 1554
 	/* Configure device */
1447
-	if ( ( rc = usb_configure_any ( usb ) ) != 0 )
1448
-		goto err_configure_any;
1555
+	if ( ( rc = usb_autoconfigure ( usb ) ) != 0 )
1556
+		goto err_autoconfigure;
1449 1557
 
1450 1558
 	return 0;
1451 1559
 
1452 1560
 	usb_deconfigure ( usb );
1453
- err_configure_any:
1561
+ err_autoconfigure:
1454 1562
  err_get_device_descriptor:
1455 1563
  err_mtu:
1456 1564
  err_get_mtu:

+ 1
- 0
src/drivers/net/dm96xx.c View File

@@ -666,6 +666,7 @@ static struct usb_device_id dm96xx_ids[] = {
666 666
 struct usb_driver dm96xx_driver __usb_driver = {
667 667
 	.ids = dm96xx_ids,
668 668
 	.id_count = ( sizeof ( dm96xx_ids ) / sizeof ( dm96xx_ids[0] ) ),
669
+	.score = USB_SCORE_NORMAL,
669 670
 	.probe = dm96xx_probe,
670 671
 	.remove = dm96xx_remove,
671 672
 };

+ 1
- 0
src/drivers/net/ecm.c View File

@@ -515,6 +515,7 @@ static struct usb_device_id ecm_ids[] = {
515 515
 struct usb_driver ecm_driver __usb_driver = {
516 516
 	.ids = ecm_ids,
517 517
 	.id_count = ( sizeof ( ecm_ids ) / sizeof ( ecm_ids[0] ) ),
518
+	.score = USB_SCORE_NORMAL,
518 519
 	.probe = ecm_probe,
519 520
 	.remove = ecm_remove,
520 521
 };

+ 1
- 0
src/drivers/net/ncm.c View File

@@ -667,6 +667,7 @@ static struct usb_device_id ncm_ids[] = {
667 667
 struct usb_driver ncm_driver __usb_driver = {
668 668
 	.ids = ncm_ids,
669 669
 	.id_count = ( sizeof ( ncm_ids ) / sizeof ( ncm_ids[0] ) ),
670
+	.score = USB_SCORE_NORMAL,
670 671
 	.probe = ncm_probe,
671 672
 	.remove = ncm_remove,
672 673
 };

+ 1
- 0
src/drivers/net/smsc75xx.c View File

@@ -1052,6 +1052,7 @@ static struct usb_device_id smsc75xx_ids[] = {
1052 1052
 struct usb_driver smsc75xx_driver __usb_driver = {
1053 1053
 	.ids = smsc75xx_ids,
1054 1054
 	.id_count = ( sizeof ( smsc75xx_ids ) / sizeof ( smsc75xx_ids[0] ) ),
1055
+	.score = USB_SCORE_NORMAL,
1055 1056
 	.probe = smsc75xx_probe,
1056 1057
 	.remove = smsc75xx_remove,
1057 1058
 };

+ 1
- 0
src/drivers/usb/usbhub.c View File

@@ -542,6 +542,7 @@ static struct usb_device_id hub_ids[] = {
542 542
 struct usb_driver usb_hub_driver __usb_driver = {
543 543
 	.ids = hub_ids,
544 544
 	.id_count = ( sizeof ( hub_ids ) / sizeof ( hub_ids[0] ) ),
545
+	.score = USB_SCORE_NORMAL,
545 546
 	.probe = hub_probe,
546 547
 	.remove = hub_remove,
547 548
 };

+ 8
- 10
src/drivers/usb/usbio.c View File

@@ -158,7 +158,7 @@ static int usbio_interface ( struct usbio_device *usbio,
158 158
 					continue;
159 159
 
160 160
 				/* Iterate over all interfaces for a match */
161
-				for ( i = 0 ; i < func->count ; i++ ) {
161
+				for ( i = 0 ; i < func->desc.count ; i++ ) {
162 162
 					if ( interface->interface ==
163 163
 					     func->interface[i] )
164 164
 						return interface->interface;
@@ -1287,15 +1287,13 @@ static int usbio_supported ( EFI_HANDLE handle ) {
1287 1287
 	EFI_BOOT_SERVICES *bs = efi_systab->BootServices;
1288 1288
 	EFI_USB_DEVICE_DESCRIPTOR device;
1289 1289
 	EFI_USB_INTERFACE_DESCRIPTOR interface;
1290
-	struct usb_class class;
1290
+	struct usb_function_descriptor desc;
1291 1291
 	struct usb_driver *driver;
1292 1292
 	struct usb_device_id *id;
1293 1293
 	union {
1294 1294
 		void *interface;
1295 1295
 		EFI_USB_IO_PROTOCOL *io;
1296 1296
 	} usb;
1297
-	unsigned int vendor;
1298
-	unsigned int product;
1299 1297
 	EFI_STATUS efirc;
1300 1298
 	int rc;
1301 1299
 
@@ -1318,8 +1316,8 @@ static int usbio_supported ( EFI_HANDLE handle ) {
1318 1316
 		       "%s\n", efi_handle_name ( handle ), strerror ( rc ) );
1319 1317
 		goto err_get_device_descriptor;
1320 1318
 	}
1321
-	vendor = device.IdVendor;
1322
-	product = device.IdProduct;
1319
+	desc.vendor = device.IdVendor;
1320
+	desc.product = device.IdProduct;
1323 1321
 
1324 1322
 	/* Get interface descriptor */
1325 1323
 	if ( ( efirc = usb.io->UsbGetInterfaceDescriptor ( usb.io,
@@ -1329,12 +1327,12 @@ static int usbio_supported ( EFI_HANDLE handle ) {
1329 1327
 		       "%s\n", efi_handle_name ( handle ), strerror ( rc ) );
1330 1328
 		goto err_get_interface_descriptor;
1331 1329
 	}
1332
-	class.class = interface.InterfaceClass;
1333
-	class.subclass = interface.InterfaceSubClass;
1334
-	class.protocol = interface.InterfaceProtocol;
1330
+	desc.class.class = interface.InterfaceClass;
1331
+	desc.class.subclass = interface.InterfaceSubClass;
1332
+	desc.class.protocol = interface.InterfaceProtocol;
1335 1333
 
1336 1334
 	/* Look for a driver for this interface */
1337
-	driver = usb_find_driver ( vendor, product, &class, &id );
1335
+	driver = usb_find_driver ( &desc, &id );
1338 1336
 	if ( ! driver ) {
1339 1337
 		rc = -ENOTSUP;
1340 1338
 		goto err_unsupported;

+ 1
- 0
src/drivers/usb/usbkbd.c View File

@@ -449,6 +449,7 @@ static struct usb_device_id usbkbd_ids[] = {
449 449
 struct usb_driver usbkbd_driver __usb_driver = {
450 450
 	.ids = usbkbd_ids,
451 451
 	.id_count = ( sizeof ( usbkbd_ids ) / sizeof ( usbkbd_ids[0] ) ),
452
+	.score = USB_SCORE_NORMAL,
452 453
 	.probe = usbkbd_probe,
453 454
 	.remove = usbkbd_remove,
454 455
 };

+ 2
- 2
src/drivers/usb/usbnet.c View File

@@ -173,7 +173,7 @@ static int usbnet_comms_describe ( struct usbnet_device *usbnet,
173 173
 	int rc;
174 174
 
175 175
 	/* Iterate over all available interfaces */
176
-	for ( i = 0 ; i < usbnet->func->count ; i++ ) {
176
+	for ( i = 0 ; i < usbnet->func->desc.count ; i++ ) {
177 177
 
178 178
 		/* Get interface number */
179 179
 		comms = usbnet->func->interface[i];
@@ -217,7 +217,7 @@ static int usbnet_data_describe ( struct usbnet_device *usbnet,
217 217
 	int rc;
218 218
 
219 219
 	/* Iterate over all available interfaces */
220
-	for ( i = 0 ; i < usbnet->func->count ; i++ ) {
220
+	for ( i = 0 ; i < usbnet->func->desc.count ; i++ ) {
221 221
 
222 222
 		/* Get interface number */
223 223
 		data = usbnet->func->interface[i];

+ 39
- 9
src/include/ipxe/usb.h View File

@@ -615,6 +615,23 @@ extern int usb_prefill ( struct usb_endpoint *ep );
615 615
 extern int usb_refill ( struct usb_endpoint *ep );
616 616
 extern void usb_flush ( struct usb_endpoint *ep );
617 617
 
618
+/**
619
+ * A USB function descriptor
620
+ *
621
+ * This is an internal descriptor used to represent an association of
622
+ * interfaces within a USB device.
623
+ */
624
+struct usb_function_descriptor {
625
+	/** Vendor ID */
626
+	uint16_t vendor;
627
+	/** Product ID */
628
+	uint16_t product;
629
+	/** Class */
630
+	struct usb_class class;
631
+	/** Number of interfaces */
632
+	unsigned int count;
633
+};
634
+
618 635
 /**
619 636
  * A USB function
620 637
  *
@@ -626,10 +643,8 @@ struct usb_function {
626 643
 	const char *name;
627 644
 	/** USB device */
628 645
 	struct usb_device *usb;
629
-	/** Class */
630
-	struct usb_class class;
631
-	/** Number of interfaces */
632
-	unsigned int count;
646
+	/** Function descriptor */
647
+	struct usb_function_descriptor desc;
633 648
 	/** Generic device */
634 649
 	struct device dev;
635 650
 	/** List of functions within this USB device */
@@ -1161,7 +1176,7 @@ usb_get_device_descriptor ( struct usb_device *usb,
1161 1176
  * @v data		Configuration descriptor to fill in
1162 1177
  * @ret rc		Return status code
1163 1178
  */
1164
-static inline __attribute (( always_inline )) int
1179
+static inline __attribute__ (( always_inline )) int
1165 1180
 usb_get_config_descriptor ( struct usb_device *usb, unsigned int index,
1166 1181
 			    struct usb_configuration_descriptor *data,
1167 1182
 			    size_t len ) {
@@ -1296,6 +1311,12 @@ struct usb_driver {
1296 1311
 	struct usb_device_id *ids;
1297 1312
 	/** Number of entries in ID table */
1298 1313
 	unsigned int id_count;
1314
+	/** Driver score
1315
+	 *
1316
+	 * This is used to determine the preferred configuration for a
1317
+	 * USB device.
1318
+	 */
1319
+	unsigned int score;
1299 1320
 	/**
1300 1321
 	 * Probe device
1301 1322
 	 *
@@ -1319,9 +1340,18 @@ struct usb_driver {
1319 1340
 /** Declare a USB driver */
1320 1341
 #define __usb_driver __table_entry ( USB_DRIVERS, 01 )
1321 1342
 
1322
-extern struct usb_driver * usb_find_driver ( unsigned int vendor,
1323
-					     unsigned int product,
1324
-					     struct usb_class *class,
1325
-					     struct usb_device_id **id );
1343
+/** USB driver scores */
1344
+enum usb_driver_score {
1345
+	/** Fallback driver (has no effect on overall score) */
1346
+	USB_SCORE_FALLBACK = 0,
1347
+	/** Deprecated driver */
1348
+	USB_SCORE_DEPRECATED = 1,
1349
+	/** Normal driver */
1350
+	USB_SCORE_NORMAL = 2,
1351
+};
1352
+
1353
+extern struct usb_driver *
1354
+usb_find_driver ( struct usb_function_descriptor *desc,
1355
+		  struct usb_device_id **id );
1326 1356
 
1327 1357
 #endif /* _IPXE_USB_H */

Loading…
Cancel
Save