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 */
T
Takashi Iwai 已提交
52 53
	int serial_number;

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

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},
M
Markus Grabner 已提交
117 118 119
	.bytes_per_frame = 4
};

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 180 181 182 183 184 185 186
	if (line6pcm->volume_monitor > 0) {
		err = line6_pcm_acquire(line6pcm, LINE6_STREAM_MONITOR);
		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 244 245 246 247
	return 1;
}

static void toneport_start_pcm(unsigned long arg)
{
	struct usb_line6_toneport *toneport = (struct usb_line6_toneport *)arg;
	struct usb_line6 *line6 = &toneport->line6;
248

249
	line6_pcm_acquire(line6->line6pcm, LINE6_STREAM_MONITOR);
250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273
}

/* control definition */
static struct snd_kcontrol_new toneport_control_monitor = {
	.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 */
static struct snd_kcontrol_new toneport_control_source = {
	.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 419 420
	setup_timer(&toneport->timer, toneport_start_pcm,
		    (unsigned long)toneport);

421 422
	line6->disconnect = line6_toneport_disconnect;

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

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

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

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

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

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

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

T
Takashi Iwai 已提交
460
#ifdef CONFIG_PM
461 462 463
/*
	Resume Toneport device after reset.
*/
T
Takashi Iwai 已提交
464
static int toneport_reset_resume(struct usb_interface *interface)
465
{
T
Takashi Iwai 已提交
466 467
	toneport_setup(usb_get_intfdata(interface));
	return line6_resume(interface);
468
}
T
Takashi Iwai 已提交
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 559
#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)
{
560
	return line6_probe(interface, id,
561
			   &toneport_properties_table[id->driver_info],
562
			   toneport_init, sizeof(struct usb_line6_toneport));
T
Takashi Iwai 已提交
563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580
}

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");