/[svn]/linuxsampler/trunk/src/drivers/midi/VirtualMidiDevice.cpp
ViewVC logotype

Annotation of /linuxsampler/trunk/src/drivers/midi/VirtualMidiDevice.cpp

Parent Directory Parent Directory | Revision Log Revision Log


Revision 2521 - (hide annotations) (download)
Wed Feb 19 19:02:43 2014 UTC (10 years, 2 months ago) by schoenebeck
File size: 6191 byte(s)
* VirtualMidiDevice: implemented support for program change, bank select
  and pitch bend.
* Bumped version (1.0.0.svn32).

1 schoenebeck 1659 /*
2 schoenebeck 2521 Copyright (C) 2008 - 2014 Christian Schoenebeck
3 schoenebeck 1659 */
4    
5     #include "VirtualMidiDevice.h"
6    
7     #include "../../common/global_private.h"
8     #include "../../common/atomic.h"
9     #include "../../common/RingBuffer.h"
10    
11 schoenebeck 2025 #define MIDI_KEYS 128
12     #define MIDI_CONTROLLERS 128
13 schoenebeck 1659
14     // assuming VirtualMidiDevice implementation is only controlled
15     // by mouse (and the user not being Billy the Kid)
16     #define MAX_EVENTS 12
17    
18     namespace LinuxSampler {
19    
20 persson 1896 struct VirtualMidiDevice::private_data_t {
21 schoenebeck 1659 atomic_t notesChanged; // whether some key changed at all
22     atomic_t pNoteChanged[MIDI_KEYS]; // which key(s) changed
23     atomic_t pNoteIsActive[MIDI_KEYS]; // status of each key (either active or inactive)
24 schoenebeck 1663 atomic_t pNoteOnVelocity[MIDI_KEYS];
25     atomic_t pNoteOffVelocity[MIDI_KEYS];
26 schoenebeck 2025 atomic_t ccsChanged; // whether some controller changed at all
27     atomic_t pCCChanged[MIDI_CONTROLLERS]; // which controller(s) changed
28     atomic_t pCCValue[MIDI_CONTROLLERS]; // current value of each controller
29 schoenebeck 1659 RingBuffer<VirtualMidiDevice::event_t,false> events;
30    
31 persson 1896 private_data_t() : events(MAX_EVENTS, 0) {}
32 schoenebeck 1659 };
33    
34 persson 1896 VirtualMidiDevice::VirtualMidiDevice() : p(new private_data_t) {
35 schoenebeck 1659 atomic_t zero = ATOMIC_INIT(0);
36 schoenebeck 1663 atomic_t defaultVelocity = ATOMIC_INIT(127);
37 schoenebeck 2025 atomic_t defaultCCValue = ATOMIC_INIT(0);
38 schoenebeck 1659 p->notesChanged = zero;
39 schoenebeck 2025 p->ccsChanged = zero;
40 schoenebeck 1659 for (int i = 0; i < MIDI_KEYS; i++) {
41     p->pNoteChanged[i] = zero;
42     p->pNoteIsActive[i] = zero;
43 schoenebeck 1663 p->pNoteOnVelocity[i] = defaultVelocity;
44     p->pNoteOffVelocity[i] = defaultVelocity;
45 schoenebeck 2025 p->pCCChanged[i] = zero;
46     p->pCCValue[i] = defaultCCValue;
47 schoenebeck 1659 }
48     }
49    
50     VirtualMidiDevice::~VirtualMidiDevice() {
51 persson 1896 delete p;
52 schoenebeck 1659 }
53 schoenebeck 2384
54     void VirtualMidiDevice::SetMaxEvents(int n) {
55     p->events.resize(n);
56     }
57 schoenebeck 1659
58     bool VirtualMidiDevice::SendNoteOnToSampler(uint8_t Key, uint8_t Velocity) {
59     if (Key >= MIDI_KEYS || Velocity > 127) return false;
60 schoenebeck 2385 if (Velocity == 0) {
61     return SendNoteOffToSampler(Key, Velocity);
62     }
63 schoenebeck 1659 event_t ev = { EVENT_TYPE_NOTEON, Key, Velocity };
64     if (p->events.write_space() <= 0) return false;
65     p->events.push(&ev);
66     return true;
67     }
68    
69     bool VirtualMidiDevice::SendNoteOffToSampler(uint8_t Key, uint8_t Velocity) {
70     if (Key >= MIDI_KEYS || Velocity > 127) return false;
71     event_t ev = { EVENT_TYPE_NOTEOFF, Key, Velocity };
72     if (p->events.write_space() <= 0) return false;
73     p->events.push(&ev);
74     return true;
75     }
76 schoenebeck 2025
77     bool VirtualMidiDevice::SendCCToSampler(uint8_t Controller, uint8_t Value) {
78     if (Controller >= MIDI_CONTROLLERS || Value > 127) return false;
79     event_t ev = { EVENT_TYPE_CC, Controller, Value };
80     if (p->events.write_space() <= 0) return false;
81     p->events.push(&ev);
82     return true;
83     }
84 schoenebeck 1659
85 schoenebeck 2521 bool VirtualMidiDevice::SendPitchBendToSampler(int Pitch) {
86     if (Pitch < -8192 || Pitch > 8191) return false;
87     Pitch += 8192;
88     // order: LSB, MSB like it would be in a regular pitch bend MIDI message
89     event_t ev = { EVENT_TYPE_PITCHBEND, Pitch & 0x7f, (Pitch >> 7) & 0x7f };
90     if (p->events.write_space() <= 0) return false;
91     p->events.push(&ev);
92     return true;
93     }
94    
95     bool VirtualMidiDevice::SendProgramChangeToSampler(int Program) {
96     if (Program < 0 || Program > 127) return false;
97     event_t ev = { EVENT_TYPE_PROGRAM, Program, 0 };
98     if (p->events.write_space() <= 0) return false;
99     p->events.push(&ev);
100     return true;
101     }
102    
103 schoenebeck 1659 bool VirtualMidiDevice::GetMidiEventFromDevice(event_t& Event) {
104     return (p->events.pop(&Event) > 0);
105     }
106    
107     bool VirtualMidiDevice::NotesChanged() {
108     int c = atomic_read( &p->notesChanged );
109     atomic_sub(c, &p->notesChanged );
110     return c;
111     }
112 schoenebeck 2025
113     bool VirtualMidiDevice::ControllersChanged() {
114     int c = atomic_read( &p->ccsChanged );
115     atomic_sub(c, &p->ccsChanged );
116     return c;
117     }
118 schoenebeck 1659
119     bool VirtualMidiDevice::NoteChanged(uint8_t Key) {
120     int c = atomic_read( &(p->pNoteChanged)[Key] );
121     atomic_sub(c, &(p->pNoteChanged)[Key] );
122     return c;
123     }
124 schoenebeck 2025
125     bool VirtualMidiDevice::ControllerChanged(uint8_t Controller) {
126     int c = atomic_read( &(p->pCCChanged)[Controller] );
127     atomic_sub(c, &(p->pCCChanged)[Controller] );
128     return c;
129     }
130 schoenebeck 1659
131     bool VirtualMidiDevice::NoteIsActive(uint8_t Key) {
132     return atomic_read( &(p->pNoteIsActive)[Key] );
133     }
134    
135 schoenebeck 1663 uint8_t VirtualMidiDevice::NoteOnVelocity(uint8_t Key) {
136     return atomic_read( &(p->pNoteOnVelocity)[Key] );
137     }
138    
139     uint8_t VirtualMidiDevice::NoteOffVelocity(uint8_t Key) {
140     return atomic_read( &(p->pNoteOffVelocity)[Key] );
141     }
142 schoenebeck 2025
143     uint8_t VirtualMidiDevice::ControllerValue(uint8_t Controller) {
144     return atomic_read( &(p->pCCValue)[Controller] );
145     }
146 schoenebeck 1663
147     void VirtualMidiDevice::SendNoteOnToDevice(uint8_t Key, uint8_t Velocity) {
148 schoenebeck 1659 if (Key >= MIDI_KEYS) return;
149 schoenebeck 2385 if (Velocity == 0) {
150     SendNoteOffToDevice(Key, Velocity);
151     return;
152     }
153 schoenebeck 1663 atomic_set( &(p->pNoteOnVelocity)[Key], Velocity );
154 schoenebeck 1659 atomic_inc( &(p->pNoteIsActive)[Key] );
155     atomic_inc( &(p->pNoteChanged)[Key] );
156     atomic_inc( &p->notesChanged );
157     }
158    
159 schoenebeck 1663 void VirtualMidiDevice::SendNoteOffToDevice(uint8_t Key, uint8_t Velocity) {
160 schoenebeck 1659 if (Key >= MIDI_KEYS) return;
161 schoenebeck 1663 atomic_set( &(p->pNoteOffVelocity)[Key], Velocity );
162 schoenebeck 2384 if (atomic_read( &(p->pNoteIsActive)[Key] )) // only decrement if not zero
163     atomic_dec( &(p->pNoteIsActive)[Key] );
164 schoenebeck 1659 atomic_inc( &(p->pNoteChanged)[Key] );
165     atomic_inc( &p->notesChanged );
166     }
167 schoenebeck 2025
168     void VirtualMidiDevice::SendCCToDevice(uint8_t Controller, uint8_t Value) {
169     if (Controller >= MIDI_CONTROLLERS) return;
170     atomic_set( &(p->pCCValue)[Controller], Value );
171     atomic_inc( &(p->pCCChanged)[Controller] );
172     atomic_inc( &p->ccsChanged );
173     }
174 schoenebeck 1659
175     } // namespace LinuxSampler

  ViewVC Help
Powered by ViewVC