Clipping means that you limit the amplitude of samples, e.g.or, observing that you might want to limit negative excursions as well as positive ones:sample_out = min(sample_in, 1.0),
sample_out = max(min(sample_in, 1.0), -1.0).This type of clipping is called hard clipping because the transition from an undistorted one to a maximum or minimum value is instantaneous, as if the signal hit a brick wall and can go no more.
As you might guess, you can also have soft clipping where the transition is more gradual, and that's what this tutorial is about.
Before discussing soft clipping, let's introduce a few other terms. Since clipping seems to be a way of limiting the values of the signal or compressing the range of the signal, you might think that clipping is a form of compressing or limiting; however, these terms have special meanings in the audio processing world.
Compression, or dynamics compression (not to be confused with data compression) is a way of reducing the overall range of soft to loud.To do dynamics compression, you generally detect the intensity or peak level of a sound over a time scale of a few milliseconds, and then construct a relatively smooth amplitude control that you apply to the signal. For compression, the amplification goes down as the intensity of the input goes up, so loud passages get (relatively) softer and soft passages get (relatively) louder. (Dynamics expansion does just the opposite.) Limiters work like compressors, but are designed to eliminate peaks -- conceptually, there is no real difference, and products are often sold as "compressor/limiters". See nyquist/lib/compress.lsp for an implementation.
In some ways, soft clipping is like dynamics compression. Both reduce the gain at high levels, but dynamics compression operates relatively slowly, effectively turning the volume knob up and down, whereas clipping operates on a sample-by-sample basis. The effect of clipping is to distort the input.
In Nyquist, you use the shape function to implement clipping. shape applies a function to each sample, where the function is specified by a sound. The following is a typical function for soft clipping:

The Nyquist shape function is allows you to specify any function you like using a sound. Therefore you can use any of the Nyquist primitives to construct the function. Since a sound is a function of time, where time must be non-negative, how do you specify a shape function over a range that includes negative values? The trick is that shape takes an offset parameter that shifts the whole function to the left (in the -y direction). Note also that whereas a Nyquist sound is generally regarded as a function of time, shape treats the sound as just a real-valued function.
The typical way to use shape is to create some increasing signal over the interval [0,2] that crosses zero at 1.0. Then you give 1.0 as the offset (3rd parameter). The result will look something like the graph above.
(setf distortion 
      (osc (hz-to-step 0.25) 2.01 *SINE-TABLE* -90.0))
(defun distort (snd)
    (shape snd distortion 1.0)) 
(play (distort (mult 15.0 (ramp 4) (osc c4 4))))
Even though I am an expert, and I have done this before, it still took me a few tries to get right. Here's a step-by-step explanation:
(s-plot (force-srate 100 distortion))
Note that the input to shape is pre-clipped to the range [-1, +1] so you only need to specify the table from [0, 2] if the origin (third parameter to shape) is 1.0. If you specify a little extra, as in this example, it will be ignored.
Look at the generated waveform to observe the soft clipping effect. Here we see the beginning of the output, where the input amplitude is low and there is very little distortion:
 
 
Generated by Roger Dannenberg (roger.dannenberg@cs.cmu.edu) Feb, 2004.