/* Renamed to PQcorktile.sl for RMR -- tal AT renderman DOT org */

/* corktile.sl - a surface shader imitating cork tiling
 *
 * Author: Peter Quint
 * Last Update: 16 October 2000
 *
 * parameters
 *
 * Ka, Kd, Ks, roughness, specularcolor - take their usual meanings
 * Ksgap - the specular component for tile gaps : by default 0.3 * Ks
 * txtscale - the amount of detail / magnification of the image, this also determines
 *   (if tiles=1) the number of tiles fitted into the texture as s and t vary from 0 to 1
 * tiles - the number of tiles in a unit square of texture before txtscale is applied
 *	 (this means the number of tiles actually seen is the square of txtscale * tiles)
 * gap - the size of the gap or groove between tiles, measured as a proportion of the tile,
 *	 thus a gap of .05 will mean that 2.5% of the tile around each edge will be taken up with
 *   the groove (2.5% either side = 5% in total = 0.05)
 * bumpheight - the total height for bump mapped bumps - irregularities on the surface plus
 *   the displacement of the tile itself out of the gap
 * tileheight - the proportion of the bumpheight that is taken up with the rise of the tile from
 * 	 the gap between the tiles, the remainder of bumpheight is taken up with the bumps on the 
 *   surface of the tile. So if tileheight = 1 there are no bumps on the tile surface, if
 *   tileheight = 0 there are bumps on the tile surface of height tileheight, but no difference
 *   in height between the gap and the tile itself
 *
 * The shader makes an attempt at being self antaliasing
 *
 */

#include "noises.h" /* the standard BMRT include file */
#include "tile.h"
#define fnoise(p,width) (noise(p) * (1-smoothstep (0.2,0.75,width)))

/* Bias function - Perlin and Hoffert 1989 */

float
bias (float b, x)
{
	return pow(x, log(b)/log(0.5));
}

color cork(		point PP;
							float txtscale, filtwidth)
	{
		/* 	the pattern is built up of the sum of two offset noise functions, perturbed slightly
		and used to select a color from a spline, combined with an overlay of the dark 
		markings that characterise cork
		*/
		color Ct;
		point P2 = transform("shader",PP);
		point PT = P2 * txtscale;
		float fw = filtwidth * txtscale;
		float g;
		float f = fnoise(PT, fw) + fnoise((P2 + 0.553) * txtscale, fw); 
		f = f * fnoise(PT * .854, fw);
		f = f + (fnoise(PT * 5.342, fw) - .5) * .5;  
		f = clamp(f, 0, 1);
		g = fBm((P2 + 8.654) * txtscale, fw, 4, 2, .5);
		Ct = color spline( 	f,
						color( 0, 0, 0),
						color( .25, 0, 0),
						color( .31, .2, 0),
						color( .486, .165, .059),
						color( .47, .18, .09),
						color( .64, .24, .07),
						color( .749, .416, .208)
						); 
		/* Overlay the dark markings */
		if (g > .2)
			{
				g = bias(0.8,(g - .2) / .8);
				Ct = Ct * (1 - g * .65);
			}
		return Ct;
	}

surface
PQcorktile( 	float	Ka = 1;
          	float	Kd = .75;
         		float	Ks = .3;
						float Ksgap = -1;
         		float	roughness = .05;
	 					float	txtscale = 1;
						float tiles = 1;
						float gap = .05;
						color specularcolor = 1;
						float bumpheight = 0.3;
						float tileheight = .8;
						float bumpfreq = 0.8;)

{
	uniform float Ksg = Ksgap > 0 ? Ksgap : Ks * .3; 
	uniform float tilewidth = 1 / (tiles * txtscale); 
	float disp, adjust;
	point PP = transform("shader",P);
	float filtwidth = filterwidthp(PP),
			swidth = abs(Du(s)*du) + abs(Dv(s)*dv),
			twidth = abs(Du(t)*du) + abs(Dv(t)*dv),
			stfiltwidth = max(swidth, twidth) / tilewidth,
			sw, tw, so, to;
	color Cg, Ct;
	normal Nf;
	float ingap = tile(s, t, tilewidth, tilewidth, swidth, twidth, gap, gap, sw, tw, so, to);
	
	Ct = cork( P, txtscale, filtwidth);	
	/* Calculate color of gap between tiles */
	Cg = color( .20, 0, 0) + filteredsnoise(PP * txtscale, filtwidth * txtscale) * .2;
	so = so > .5 ? 1 - so : so;
	to = to > .5 ? 1 - to : to; 
	adjust = (1 - min(smoothstep(gap * .3, gap * .7, so), smoothstep(gap * .3, gap * .7, to))) * 
		(1  - smoothstep(.2 * gap, gap * .8, stfiltwidth));  
	disp = (tileheight + fnoise(PP * txtscale * bumpfreq + 3, filtwidth * txtscale) * (1 - tileheight)) 
		* bumpheight * (1 - adjust);
	Nf = faceforward(calculatenormal(P + disp *	normalize(N)), I);
	Nf = normalize(Nf);
	/* printf("Nf = %p\n", Nf);	 */
	Oi = Os;
	Ci = Os * (mix( Ct, Cg, ingap) * (Ka*ambient() + Kd*diffuse(Nf)) +
		specularcolor * Ks*specular(Nf,-normalize(I),roughness)); 
	}

