toneport.c 14.2 KB
Newer Older
M
Markus Grabner 已提交
1
/*
2
 * Line 6 Linux USB driver
M
Markus Grabner 已提交
3
 *
4
 * Copyright (C) 2004-2010 Markus Grabner (grabner@icg.tugraz.at)
M
Markus Grabner 已提交
5 6 7 8 9 10 11 12
 *                         Emil Myhrman (emil.myhrman@gmail.com)
 *
 *	This program is free software; you can redistribute it and/or
 *	modify it under the terms of the GNU General Public License as
 *	published by the Free Software Foundation, version 2.
 *
 */

13
#include <linux/wait.h>
T
Takashi Iwai 已提交
14 15 16
#include <linux/usb.h>
#include <linux/slab.h>
#include <linux/module.h>
17
#include <linux/leds.h>
T
Takashi Iwai 已提交
18
#include <sound/core.h>
19
#include <sound/control.h>
M
Markus Grabner 已提交
20 21

#include "capture.h"
22
#include "driver.h"
M
Markus Grabner 已提交
23
#include "playback.h"
T
Takashi Iwai 已提交
24 25 26 27 28 29 30 31 32 33 34

enum line6_device_type {
	LINE6_GUITARPORT,
	LINE6_PODSTUDIO_GX,
	LINE6_PODSTUDIO_UX1,
	LINE6_PODSTUDIO_UX2,
	LINE6_TONEPORT_GX,
	LINE6_TONEPORT_UX1,
	LINE6_TONEPORT_UX2,
};

35 36 37 38 39 40 41 42 43
struct usb_line6_toneport;

struct toneport_led {
	struct led_classdev dev;
	char name[64];
	struct usb_line6_toneport *toneport;
	bool registered;
};

T
Takashi Iwai 已提交
44
struct usb_line6_toneport {
45
	/* Generic Line 6 USB data */
T
Takashi Iwai 已提交
46 47
	struct usb_line6 line6;

48
	/* Source selector */
T
Takashi Iwai 已提交
49 50
	int source;

51
	/* Serial number of device */
52
	u32 serial_number;
T
Takashi Iwai 已提交
53

54
	/* Firmware version (x 100) */
55
	u8 firmware_version;
T
Takashi Iwai 已提交
56

57
	/* Timer for delayed PCM startup */
T
Takashi Iwai 已提交
58 59
	struct timer_list timer;

60
	/* Device type */
T
Takashi Iwai 已提交
61
	enum line6_device_type type;
62 63 64

	/* LED instances */
	struct toneport_led leds[2];
T
Takashi Iwai 已提交
65
};
M
Markus Grabner 已提交
66 67 68

static int toneport_send_cmd(struct usb_device *usbdev, int cmd1, int cmd2);

69 70
#define TONEPORT_PCM_DELAY 1

M
Markus Grabner 已提交
71 72 73 74 75 76 77 78
static struct snd_ratden toneport_ratden = {
	.num_min = 44100,
	.num_max = 44100,
	.num_step = 1,
	.den = 1
};

static struct line6_pcm_properties toneport_pcm_properties = {
79
	.playback_hw = {
80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96
				  .info = (SNDRV_PCM_INFO_MMAP |
					   SNDRV_PCM_INFO_INTERLEAVED |
					   SNDRV_PCM_INFO_BLOCK_TRANSFER |
					   SNDRV_PCM_INFO_MMAP_VALID |
					   SNDRV_PCM_INFO_PAUSE |
					   SNDRV_PCM_INFO_SYNC_START),
				  .formats = SNDRV_PCM_FMTBIT_S16_LE,
				  .rates = SNDRV_PCM_RATE_KNOT,
				  .rate_min = 44100,
				  .rate_max = 44100,
				  .channels_min = 2,
				  .channels_max = 2,
				  .buffer_bytes_max = 60000,
				  .period_bytes_min = 64,
				  .period_bytes_max = 8192,
				  .periods_min = 1,
				  .periods_max = 1024},
97
	.capture_hw = {
98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113
				 .info = (SNDRV_PCM_INFO_MMAP |
					  SNDRV_PCM_INFO_INTERLEAVED |
					  SNDRV_PCM_INFO_BLOCK_TRANSFER |
					  SNDRV_PCM_INFO_MMAP_VALID |
					  SNDRV_PCM_INFO_SYNC_START),
				 .formats = SNDRV_PCM_FMTBIT_S16_LE,
				 .rates = SNDRV_PCM_RATE_KNOT,
				 .rate_min = 44100,
				 .rate_max = 44100,
				 .channels_min = 2,
				 .channels_max = 2,
				 .buffer_bytes_max = 60000,
				 .period_bytes_min = 64,
				 .period_bytes_max = 8192,
				 .periods_min = 1,
				 .periods_max = 1024},
114
	.rates = {
115 116
			    .nrats = 1,
			    .rats = &toneport_ratden},
117
	.bytes_per_channel = 2
M
Markus Grabner 已提交
118 119
};

120
static const struct {
121 122
	const char *name;
	int code;
123
} toneport_source_info[] = {
124 125 126 127
	{"Microphone", 0x0a01},
	{"Line", 0x0801},
	{"Instrument", 0x0b01},
	{"Inst & Mic", 0x0901}
128 129
};

M
Markus Grabner 已提交
130 131 132 133
static int toneport_send_cmd(struct usb_device *usbdev, int cmd1, int cmd2)
{
	int ret;

134 135 136 137 138
	ret = usb_control_msg(usbdev, usb_sndctrlpipe(usbdev, 0), 0x67,
			      USB_TYPE_VENDOR | USB_RECIP_DEVICE | USB_DIR_OUT,
			      cmd1, cmd2, NULL, 0, LINE6_TIMEOUT * HZ);

	if (ret < 0) {
139
		dev_err(&usbdev->dev, "send failed (error %d)\n", ret);
M
Markus Grabner 已提交
140 141 142 143 144 145
		return ret;
	}

	return 0;
}

146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161
/* monitor info callback */
static int snd_toneport_monitor_info(struct snd_kcontrol *kcontrol,
				     struct snd_ctl_elem_info *uinfo)
{
	uinfo->type = SNDRV_CTL_ELEM_TYPE_INTEGER;
	uinfo->count = 1;
	uinfo->value.integer.min = 0;
	uinfo->value.integer.max = 256;
	return 0;
}

/* monitor get callback */
static int snd_toneport_monitor_get(struct snd_kcontrol *kcontrol,
				    struct snd_ctl_elem_value *ucontrol)
{
	struct snd_line6_pcm *line6pcm = snd_kcontrol_chip(kcontrol);
162

163 164 165 166 167 168 169 170 171
	ucontrol->value.integer.value[0] = line6pcm->volume_monitor;
	return 0;
}

/* monitor put callback */
static int snd_toneport_monitor_put(struct snd_kcontrol *kcontrol,
				    struct snd_ctl_elem_value *ucontrol)
{
	struct snd_line6_pcm *line6pcm = snd_kcontrol_chip(kcontrol);
172
	int err;
173

174
	if (ucontrol->value.integer.value[0] == line6pcm->volume_monitor)
175 176 177
		return 0;

	line6pcm->volume_monitor = ucontrol->value.integer.value[0];
178

179
	if (line6pcm->volume_monitor > 0) {
180
		err = line6_pcm_acquire(line6pcm, LINE6_STREAM_MONITOR, true);
181 182 183 184 185 186
		if (err < 0) {
			line6pcm->volume_monitor = 0;
			line6_pcm_release(line6pcm, LINE6_STREAM_MONITOR);
			return err;
		}
	} else {
187
		line6_pcm_release(line6pcm, LINE6_STREAM_MONITOR);
188
	}
189

190 191 192 193 194 195 196 197
	return 1;
}

/* source info callback */
static int snd_toneport_source_info(struct snd_kcontrol *kcontrol,
				    struct snd_ctl_elem_info *uinfo)
{
	const int size = ARRAY_SIZE(toneport_source_info);
198

199 200 201 202
	uinfo->type = SNDRV_CTL_ELEM_TYPE_ENUMERATED;
	uinfo->count = 1;
	uinfo->value.enumerated.items = size;

203
	if (uinfo->value.enumerated.item >= size)
204 205 206 207 208 209 210 211 212 213 214 215 216
		uinfo->value.enumerated.item = size - 1;

	strcpy(uinfo->value.enumerated.name,
	       toneport_source_info[uinfo->value.enumerated.item].name);

	return 0;
}

/* source get callback */
static int snd_toneport_source_get(struct snd_kcontrol *kcontrol,
				   struct snd_ctl_elem_value *ucontrol)
{
	struct snd_line6_pcm *line6pcm = snd_kcontrol_chip(kcontrol);
217 218
	struct usb_line6_toneport *toneport =
	    (struct usb_line6_toneport *)line6pcm->line6;
219 220 221 222 223 224 225 226 227
	ucontrol->value.enumerated.item[0] = toneport->source;
	return 0;
}

/* source put callback */
static int snd_toneport_source_put(struct snd_kcontrol *kcontrol,
				   struct snd_ctl_elem_value *ucontrol)
{
	struct snd_line6_pcm *line6pcm = snd_kcontrol_chip(kcontrol);
228 229
	struct usb_line6_toneport *toneport =
	    (struct usb_line6_toneport *)line6pcm->line6;
230
	unsigned int source;
231

232 233 234 235
	source = ucontrol->value.enumerated.item[0];
	if (source >= ARRAY_SIZE(toneport_source_info))
		return -EINVAL;
	if (source == toneport->source)
236 237
		return 0;

238
	toneport->source = source;
239
	toneport_send_cmd(toneport->line6.usbdev,
240
			  toneport_source_info[source].code, 0x0000);
241 242 243
	return 1;
}

244
static void toneport_start_pcm(struct timer_list *t)
245
{
246
	struct usb_line6_toneport *toneport = from_timer(toneport, t, timer);
247
	struct usb_line6 *line6 = &toneport->line6;
248

249
	line6_pcm_acquire(line6->line6pcm, LINE6_STREAM_MONITOR, true);
250 251 252
}

/* control definition */
253
static const struct snd_kcontrol_new toneport_control_monitor = {
254 255 256 257 258 259 260 261 262 263
	.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
	.name = "Monitor Playback Volume",
	.index = 0,
	.access = SNDRV_CTL_ELEM_ACCESS_READWRITE,
	.info = snd_toneport_monitor_info,
	.get = snd_toneport_monitor_get,
	.put = snd_toneport_monitor_put
};

/* source selector definition */
264
static const struct snd_kcontrol_new toneport_control_source = {
265 266 267 268 269 270 271 272 273
	.iface = SNDRV_CTL_ELEM_IFACE_MIXER,
	.name = "PCM Capture Source",
	.index = 0,
	.access = SNDRV_CTL_ELEM_ACCESS_READWRITE,
	.info = snd_toneport_source_info,
	.get = snd_toneport_source_get,
	.put = snd_toneport_source_put
};

274 275 276 277 278 279 280
/*
	For the led on Guitarport.
	Brightness goes from 0x00 to 0x26. Set a value above this to have led
	blink.
	(void cmd_0x02(byte red, byte green)
*/

281
static bool toneport_has_led(struct usb_line6_toneport *toneport)
282
{
283 284 285
	switch (toneport->type) {
	case LINE6_GUITARPORT:
	case LINE6_TONEPORT_GX:
286
	/* add your device here if you are missing support for the LEDs */
287 288 289 290 291
		return true;

	default:
		return false;
	}
292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350
}

static const char * const led_colors[2] = { "red", "green" };
static const int led_init_vals[2] = { 0x00, 0x26 };

static void toneport_update_led(struct usb_line6_toneport *toneport)
{
	toneport_send_cmd(toneport->line6.usbdev,
			  (toneport->leds[0].dev.brightness << 8) | 0x0002,
			  toneport->leds[1].dev.brightness);
}

static void toneport_led_brightness_set(struct led_classdev *led_cdev,
					enum led_brightness brightness)
{
	struct toneport_led *leds =
		container_of(led_cdev, struct toneport_led, dev);
	toneport_update_led(leds->toneport);
}

static int toneport_init_leds(struct usb_line6_toneport *toneport)
{
	struct device *dev = &toneport->line6.usbdev->dev;
	int i, err;

	for (i = 0; i < 2; i++) {
		struct toneport_led *led = &toneport->leds[i];
		struct led_classdev *leddev = &led->dev;

		led->toneport = toneport;
		snprintf(led->name, sizeof(led->name), "%s::%s",
			 dev_name(dev), led_colors[i]);
		leddev->name = led->name;
		leddev->brightness = led_init_vals[i];
		leddev->max_brightness = 0x26;
		leddev->brightness_set = toneport_led_brightness_set;
		err = led_classdev_register(dev, leddev);
		if (err)
			return err;
		led->registered = true;
	}

	return 0;
}

static void toneport_remove_leds(struct usb_line6_toneport *toneport)
{
	struct toneport_led *led;
	int i;

	for (i = 0; i < 2; i++) {
		led = &toneport->leds[i];
		if (!led->registered)
			break;
		led_classdev_unregister(&led->dev);
		led->registered = false;
	}
}

351 352 353 354 355 356 357 358 359 360 361 362 363 364
static bool toneport_has_source_select(struct usb_line6_toneport *toneport)
{
	switch (toneport->type) {
	case LINE6_TONEPORT_UX1:
	case LINE6_TONEPORT_UX2:
	case LINE6_PODSTUDIO_UX1:
	case LINE6_PODSTUDIO_UX2:
		return true;

	default:
		return false;
	}
}

M
Markus Grabner 已提交
365
/*
366
	Setup Toneport device.
M
Markus Grabner 已提交
367
*/
368
static void toneport_setup(struct usb_line6_toneport *toneport)
M
Markus Grabner 已提交
369
{
370
	int ticks;
M
Markus Grabner 已提交
371
	struct usb_line6 *line6 = &toneport->line6;
372 373 374 375 376 377 378 379 380 381
	struct usb_device *usbdev = line6->usbdev;

	/* sync time on device with host: */
	ticks = (int)get_seconds();
	line6_write_data(line6, 0x80c6, &ticks, 4);

	/* enable device: */
	toneport_send_cmd(usbdev, 0x0301, 0x0000);

	/* initialize source select: */
382
	if (toneport_has_source_select(toneport))
383 384 385
		toneport_send_cmd(usbdev,
				  toneport_source_info[toneport->source].code,
				  0x0000);
386

387
	if (toneport_has_led(toneport))
388
		toneport_update_led(toneport);
389 390

	mod_timer(&toneport->timer, jiffies + TONEPORT_PCM_DELAY * HZ);
391 392
}

393 394 395
/*
	Toneport device disconnected.
*/
396
static void line6_toneport_disconnect(struct usb_line6 *line6)
397
{
398 399
	struct usb_line6_toneport *toneport =
		(struct usb_line6_toneport *)line6;
400 401 402

	del_timer_sync(&toneport->timer);

403
	if (toneport_has_led(toneport))
404
		toneport_remove_leds(toneport);
405 406 407
}


408 409 410
/*
	 Try to init Toneport device.
*/
411 412
static int toneport_init(struct usb_line6 *line6,
			 const struct usb_device_id *id)
413 414
{
	int err;
415
	struct usb_line6_toneport *toneport =  (struct usb_line6_toneport *) line6;
M
Markus Grabner 已提交
416

417
	toneport->type = id->driver_info;
418
	timer_setup(&toneport->timer, toneport_start_pcm, 0);
419

420 421
	line6->disconnect = line6_toneport_disconnect;

M
Markus Grabner 已提交
422
	/* initialize PCM subsystem: */
423
	err = line6_init_pcm(line6, &toneport_pcm_properties);
424
	if (err < 0)
M
Markus Grabner 已提交
425 426
		return err;

427
	/* register monitor control: */
428 429 430 431
	err = snd_ctl_add(line6->card,
			  snd_ctl_new1(&toneport_control_monitor,
				       line6->line6pcm));
	if (err < 0)
432 433 434
		return err;

	/* register source select control: */
435
	if (toneport_has_source_select(toneport)) {
436 437 438 439
		err =
		    snd_ctl_add(line6->card,
				snd_ctl_new1(&toneport_control_source,
					     line6->line6pcm));
440
		if (err < 0)
441 442 443
			return err;
	}

M
Markus Grabner 已提交
444 445 446
	line6_read_serial_number(line6, &toneport->serial_number);
	line6_read_data(line6, 0x80c2, &toneport->firmware_version, 1);

447
	if (toneport_has_led(toneport)) {
448
		err = toneport_init_leds(toneport);
449 450
		if (err < 0)
			return err;
451
	}
M
Markus Grabner 已提交
452

453
	toneport_setup(toneport);
M
Markus Grabner 已提交
454

455 456
	/* register audio system: */
	return snd_card_register(line6->card);
457 458
}

T
Takashi Iwai 已提交
459
#ifdef CONFIG_PM
460 461 462
/*
	Resume Toneport device after reset.
*/
T
Takashi Iwai 已提交
463
static int toneport_reset_resume(struct usb_interface *interface)
464
{
T
Takashi Iwai 已提交
465 466
	toneport_setup(usb_get_intfdata(interface));
	return line6_resume(interface);
467
}
T
Takashi Iwai 已提交
468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558
#endif

#define LINE6_DEVICE(prod) USB_DEVICE(0x0e41, prod)
#define LINE6_IF_NUM(prod, n) USB_DEVICE_INTERFACE_NUMBER(0x0e41, prod, n)

/* table of devices that work with this driver */
static const struct usb_device_id toneport_id_table[] = {
	{ LINE6_DEVICE(0x4750),    .driver_info = LINE6_GUITARPORT },
	{ LINE6_DEVICE(0x4153),    .driver_info = LINE6_PODSTUDIO_GX },
	{ LINE6_DEVICE(0x4150),    .driver_info = LINE6_PODSTUDIO_UX1 },
	{ LINE6_IF_NUM(0x4151, 0), .driver_info = LINE6_PODSTUDIO_UX2 },
	{ LINE6_DEVICE(0x4147),    .driver_info = LINE6_TONEPORT_GX },
	{ LINE6_DEVICE(0x4141),    .driver_info = LINE6_TONEPORT_UX1 },
	{ LINE6_IF_NUM(0x4142, 0), .driver_info = LINE6_TONEPORT_UX2 },
	{}
};

MODULE_DEVICE_TABLE(usb, toneport_id_table);

static const struct line6_properties toneport_properties_table[] = {
	[LINE6_GUITARPORT] = {
		.id = "GuitarPort",
		.name = "GuitarPort",
		.capabilities	= LINE6_CAP_PCM,
		.altsetting = 2,  /* 1..4 seem to be ok */
		/* no control channel */
		.ep_audio_r = 0x82,
		.ep_audio_w = 0x01,
	},
	[LINE6_PODSTUDIO_GX] = {
		.id = "PODStudioGX",
		.name = "POD Studio GX",
		.capabilities	= LINE6_CAP_PCM,
		.altsetting = 2,  /* 1..4 seem to be ok */
		/* no control channel */
		.ep_audio_r = 0x82,
		.ep_audio_w = 0x01,
	},
	[LINE6_PODSTUDIO_UX1] = {
		.id = "PODStudioUX1",
		.name = "POD Studio UX1",
		.capabilities	= LINE6_CAP_PCM,
		.altsetting = 2,  /* 1..4 seem to be ok */
		/* no control channel */
		.ep_audio_r = 0x82,
		.ep_audio_w = 0x01,
	},
	[LINE6_PODSTUDIO_UX2] = {
		.id = "PODStudioUX2",
		.name = "POD Studio UX2",
		.capabilities	= LINE6_CAP_PCM,
		.altsetting = 2,  /* defaults to 44.1kHz, 16-bit */
		/* no control channel */
		.ep_audio_r = 0x82,
		.ep_audio_w = 0x01,
	},
	[LINE6_TONEPORT_GX] = {
		.id = "TonePortGX",
		.name = "TonePort GX",
		.capabilities	= LINE6_CAP_PCM,
		.altsetting = 2,  /* 1..4 seem to be ok */
		/* no control channel */
		.ep_audio_r = 0x82,
		.ep_audio_w = 0x01,
	},
	[LINE6_TONEPORT_UX1] = {
		.id = "TonePortUX1",
		.name = "TonePort UX1",
		.capabilities	= LINE6_CAP_PCM,
		.altsetting = 2,  /* 1..4 seem to be ok */
		/* no control channel */
		.ep_audio_r = 0x82,
		.ep_audio_w = 0x01,
	},
	[LINE6_TONEPORT_UX2] = {
		.id = "TonePortUX2",
		.name = "TonePort UX2",
		.capabilities	= LINE6_CAP_PCM,
		.altsetting = 2,  /* defaults to 44.1kHz, 16-bit */
		/* no control channel */
		.ep_audio_r = 0x82,
		.ep_audio_w = 0x01,
	},
};

/*
	Probe USB device.
*/
static int toneport_probe(struct usb_interface *interface,
			  const struct usb_device_id *id)
{
559
	return line6_probe(interface, id, "Line6-TonePort",
560
			   &toneport_properties_table[id->driver_info],
561
			   toneport_init, sizeof(struct usb_line6_toneport));
T
Takashi Iwai 已提交
562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579
}

static struct usb_driver toneport_driver = {
	.name = KBUILD_MODNAME,
	.probe = toneport_probe,
	.disconnect = line6_disconnect,
#ifdef CONFIG_PM
	.suspend = line6_suspend,
	.resume = line6_resume,
	.reset_resume = toneport_reset_resume,
#endif
	.id_table = toneport_id_table,
};

module_usb_driver(toneport_driver);

MODULE_DESCRIPTION("TonePort USB driver");
MODULE_LICENSE("GPL");