tutorials:basic-spectral-light-modeling
Differences
This shows you the differences between two versions of the page.
Both sides previous revisionPrevious revisionNext revision | Previous revision | ||
tutorials:basic-spectral-light-modeling [2024/12/02 09:57] – MH | tutorials:basic-spectral-light-modeling [2025/01/08 21:30] (current) – MH | ||
---|---|---|---|
Line 1: | Line 1: | ||
====== Spectral Light Modelling ====== | ====== Spectral Light Modelling ====== | ||
- | ===== General Introduction ===== | + | These three core aspects of light simulation—global and local illumination models, and light sources—are the base for any light simulation. When it comes to spectral light simulations, |
- | Light modelling generally involves three aspects: | + | Note: The hardware requirement for performing GPU-based |
- | + | ||
- | * Global illumination model | + | |
- | * Light sources | + | |
- | * Local illumination model | + | |
- | + | ||
- | {{ : | + | |
- | + | ||
- | Whereas the Global illumination model handles the actual light computation, | + | |
- | + | ||
- | In each aspect, computer graphics offers plenty of alternatives. | + | |
- | + | ||
- | {{ : | + | |
- | + | ||
- | Several of them are implemented in GroIMP as ready-to-use tools. | + | |
- | + | ||
- | GroIMP integrates two two main **light model implementations**, | + | |
- | + | ||
- | * Twilight, a CPU-based | + | |
- | * GPUFlux, | + | |
- | + | ||
- | Both implementing different global illumination model for rendering and for light computation. | + | |
- | + | ||
- | {{ : | + | |
- | + | ||
- | In the following, only light computation or light modelling | + | |
- | + | ||
- | + | ||
- | Regarding light sources, GroIMP provides a complete set of possible implementations. They all implement the Light and LightBase interfaces, which makes them easy to handle and exchange. | + | |
- | + | ||
- | {{ : | + | |
- | + | ||
- | For the Local illumination model, which defines the optical properties of the scene objects such as values for absorption, transmission, | + | |
- | + | ||
- | {{ : | + | |
- | + | ||
- | GroIMP provides a set of standard shader implementations, | + | |
- | + | ||
- | {{ : | + | |
- | + | ||
- | + | ||
- | ===== Spectral light modelling ===== | + | |
- | + | ||
- | + | ||
- | These three core aspects | + | |
+ | {{ : | ||
Spectral light simulations now deal not only with the pure calculation of light distributions but also include aspects of the principal characteristics of light, i.e., light quality, quantity, and duration. | Spectral light simulations now deal not only with the pure calculation of light distributions but also include aspects of the principal characteristics of light, i.e., light quality, quantity, and duration. | ||
Line 62: | Line 19: | ||
In GroIMP, the GPUFlux model allows us to simulate spectral light between 380 and 720 nm (default values). | In GroIMP, the GPUFlux model allows us to simulate spectral light between 380 and 720 nm (default values). | ||
- | Note: The implementation | + | Note: The implementation |
- | The spectral range [minLambda, maxLambda] can be divided into an user-defined number of equal sized so-called buckets | + | The spectral range //[minLambda, maxLambda]// can be divided into a user-defined number of equally-sized so-called buckets—sub-channels. They can be one, taking the whole range as one channel—which, in my opinion, would not make much sense—or can be as large as the number of integer wavelengths of the range, leading to 1 nm buckets if desired. Common numbers of buckets |
{{ : | {{ : | ||
- | Setting up the GPUFlux light model within GroIMP, or better | + | Setting up the GPUFlux light model within GroIMP, or more accurately |
+ | |||
+ | Note: The examples require GroIPM version >=2.0 to run. With GroIMP version 2.0 some changes on the internal package structure are made. formally classes found in de.grogra.imp3d have been moved to de.grogra.gpuflux.imp3d to match the package name (Java 11 forbid package name split). So, if you are using objects, lights or shaders from gpuflux, they should be imported as de.grogra.gpuflux.imp3d.xxx. | ||
+ | |||
+ | <code java> | ||
+ | import de.grogra.imp3d.spectral.IrregularSpectralCurve; | ||
+ | import de.grogra.gpuflux.imp3d.spectral.IrregularSpectralCurve; | ||
+ | |||
+ | // Light nodes need to be imported like this | ||
+ | import de.grogra.gpuflux.imp3d.objects.PhysicalLight; | ||
+ | </ | ||
<code java> | <code java> | ||
- | import de.grogra.imp3d.spectral.IrregularSpectralCurve; | ||
- | import de.grogra.ray.physics.Spectrum; | ||
import de.grogra.gpuflux.tracer.FluxLightModelTracer.MeasureMode; | import de.grogra.gpuflux.tracer.FluxLightModelTracer.MeasureMode; | ||
- | import de.grogra.gpuflux.scene.experiment.Measurement; | ||
... | ... | ||
Line 82: | Line 46: | ||
protected void calculateLight() { | protected void calculateLight() { | ||
- | |||
LM.setMeasureMode(MeasureMode.FULL_SPECTRUM); | LM.setMeasureMode(MeasureMode.FULL_SPECTRUM); | ||
LM.setSpectralBuckets(21); | LM.setSpectralBuckets(21); | ||
Line 89: | Line 52: | ||
</ | </ | ||
- | The GPUFlux light model supports three different modes of measuring spectral power: | + | The GPUFlux light model supports three different modes of measuring spectral power: |
- | * regular RGB, that simulates only three buckets approximating the three colour channels | + | * regular RGB, which simulates only three buckets approximating the three colour channels |
- | * full discretized spectral measurements, | + | * fully discretized spectral measurements, |
* weighted integration, | * weighted integration, | ||
Line 103: | Line 66: | ||
// disables the simulation of sensor | // disables the simulation of sensor | ||
- | LM.setEnableSensors(false); // default: | + | LM.setEnableSensors(true); // default: |
// sets the random seed for the random number generator; us this to obtain reproducible results | // sets the random seed for the random number generator; us this to obtain reproducible results | ||
LM.setRandomseed(123456); | LM.setRandomseed(123456); | ||
- | // disables | + | // enable |
- | LM.setDispersion(false); // default: | + | LM.setDispersion(ture); // default: |
</ | </ | ||
- | After the light model is configured, it can be invoked by calling the compute() function as following: | + | After the light model is configured, it can be invoked by calling the //compute()// function as follows: |
<code java> | <code java> | ||
Line 119: | Line 82: | ||
</ | </ | ||
- | To obtain the total abound | + | To obtain the total amount |
<code java> | <code java> | ||
Line 126: | Line 89: | ||
</ | </ | ||
- | By doing this within a rule, the light absorption can be obtained for all objects of the specified type, e.g., Box as in this example: | + | By doing this within a rule, the light absorption can be obtained for all objects of the specified type, such as a //Box//, as in this example: |
<code java> | <code java> | ||
Line 137: | Line 100: | ||
</ | </ | ||
- | Accessing the absorption values for each bucket can be done by accessing the data variable of the Measurement class. | + | Accessing the absorption values for each bucket can be done by accessing the data variable of the //Measurement// class. |
<code java> | <code java> | ||
Line 156: | Line 119: | ||
==== Light sources ==== | ==== Light sources ==== | ||
- | + | After the light model is set up, the next step is to define | |
- | After the light model is set-up, the next thing to do is the definition of spectral light sources. The GPUFlux light model works with all basic light nodes, | + | |
<code java> | <code java> | ||
Line 166: | Line 128: | ||
{{ : | {{ : | ||
- | will lead to some dark magenta. | ||
- | Note: The step size does not to be equal and values in-between are linear | + | Note: The step size does not have to be equal, and values in between are linearly |
- | A spectra, given by an array of wavelengths and corresponding amplitudes is called spectral curve and in computer graphics it defines a spectral power distribution. In GroIMP a spectral curve and be defined using the IrregularSpectralCurve class, which takes the wavelength array and the corresponding amplitudes as input. The IrregularSpectralCurve can be used as input to the ChannelSPD class so that it can be used later as input for the light node. | + | A spectrum, given by an array of wavelengths and corresponding amplitudes, is called |
<code java> | <code java> | ||
Line 179: | Line 140: | ||
</ | </ | ||
- | Beside | + | Besides |
* BlackbodySpectralCurve | * BlackbodySpectralCurve | ||
Line 189: | Line 150: | ||
{{ : | {{ : | ||
- | Since these spectral curve classes | + | |
+ | Since these spectral curve classes all implement | ||
<code java> | <code java> | ||
+ | //user defined spectral curve, applied to an IrregularSpectralCurve | ||
float[] WAVELENGTHS = {380, 485, 490, 610, 615, 720}; | float[] WAVELENGTHS = {380, 485, 490, 610, 615, 720}; | ||
float[] AMPLITUDES = {0, | float[] AMPLITUDES = {0, | ||
ChannelSPD GREEN_SPD = new ChannelSPD(new IrregularSpectralCurve(WAVELENGTHS, | ChannelSPD GREEN_SPD = new ChannelSPD(new IrregularSpectralCurve(WAVELENGTHS, | ||
- | ChannelSPD GREEN_SPD = new ChannelSPD(new RGBSpectralCurve(1, | + | // |
+ | ChannelSPD GREEN_SPD = new ChannelSPD(new RGBSpectralCurve(0, | ||
+ | |||
+ | //a constant spectral curve of the intensity of 0.25 equally over the whole spectrum | ||
ChannelSPD CONST_SPD = new ChannelSPD(new ConstantSpectralCurve(0.25)); | ChannelSPD CONST_SPD = new ChannelSPD(new ConstantSpectralCurve(0.25)); | ||
+ | |||
+ | //a regular spectral curve will apply the given intensities over the specified range ([400,700]) | ||
ChannelSPD REG_SPD = new ChannelSPD(new | ChannelSPD REG_SPD = new ChannelSPD(new | ||
+ | |||
+ | //a CIE Norm D55 spectral curve - sun light | ||
ChannelSPD REG_SPD = new ChannelSPD(new CIENormSpectralCurve(Attributes.CIE_NORM_D55)); | ChannelSPD REG_SPD = new ChannelSPD(new CIENormSpectralCurve(Attributes.CIE_NORM_D55)); | ||
+ | |||
+ | // a black body spectral curve with a temperature of 5000K | ||
ChannelSPD REG_SPD = new ChannelSPD(new BlackbodySpectralCurve(5000)); | ChannelSPD REG_SPD = new ChannelSPD(new BlackbodySpectralCurve(5000)); | ||
</ | </ | ||
Line 205: | Line 177: | ||
{{ : | {{ : | ||
- | To use the spectral curve as input for a light source, a SpectralLight needs to be defined. | + | To use the spectral curve as input for a light source, a //SpectralLight// needs to be defined. |
<code java> | <code java> | ||
Line 224: | Line 196: | ||
- | To complete the definition of a light source, | + | To complete the definition of a light source, |
- | The physical light distribution can be defined as a polar distribution diagram (also called polar curve) showing the luminous intensity values with increasing angles from two imaginary axes of the lamp which is placed in the centre. Red: 0–180◦ plane, blue 90–270◦ plane. On the right of the Figure below, a 3D visualisation of the same light source is given. The colour of each point (gradient from black to bright red) as well as the distance to the light source both indicate the power emitted by a light source | + | The physical light distribution can be defined as a polar distribution diagram (also called polar curve) showing the luminous intensity values with increasing angles from two imaginary axes of the lamp which is placed in the centre. Red: 0–180◦ plane, blue 90–270◦ plane. On the right of the Figure below, a 3D visualisation of the same light source is given. The colour of each point (gradient from black to bright red), as well as the distance to the light source, both indicate the power emitted by the light source in a particular direction per unit solid angle. |
- | in a particular direction per unit solid angle. | + | |
{{ : | {{ : | ||
- | Within GroIMP the PLD can be visualized for any light source as illustrated below for a SpotLight. To activate the light ray visualization, | + | Within GroIMP, the PLD can be visualized for any light source, as illustrated below for a //SpotLight//. To activate the light ray visualization, |
<code java> | <code java> | ||
Line 247: | Line 218: | ||
</ | </ | ||
- | The result of the light ray visualization, | + | The result of the light ray visualization, |
- | light, with a defined opening angle; b) user defined distribution; | + | |
- | distributed; | + | |
{{ : | {{ : | ||
- | To see the more realistic light pattern, the scene needs to be rendered | + | To see a more realistic light pattern, the scene needs to be rendered |
{{ : | {{ : | ||
- | Defining a PLD for a light source can be done in two ways: 1) ' | + | Defining a PLD for a light source can be done in two ways: 1) ' |
- | In any case, instead of one of the predefined light sources, | + | In any case, instead of one of the predefined light sources, |
Line 288: | Line 257: | ||
==== Local illumination - Shader ==== | ==== Local illumination - Shader ==== | ||
+ | |||
+ | After the definition of the global illumination model and the light sources, the last missing part required for proper spectral light modelling is the definition of the local illumination model. In computer graphics, the tools used are called Shaders. A shader defines the local optical properties of an object, namely the values for reflection, absorption, and transmission. The Phong illumination model, or Phong shader for short, allows us to define all required aspects. | ||
+ | |||
+ | In the same way as the spectral curves are defined for the light sources, the spectrum for reflectance and transmission needs to be defined for our spectral shader. The values for absorption are obtained as the ' | ||
+ | |||
+ | Note: there is no check of plausibility implemented within the Phong shader. The user needs to make sure that the sum of reflectance and transmission is not higher than the actual incoming radiation. You cannot reflect or transmit more than what was incoming; otherwise, the object would be a light source emitting light. | ||
+ | |||
+ | A Phong shader can be defined as following: | ||
+ | |||
+ | <code java> | ||
+ | float[] WAVELENGTHS = {380, 485, 490, 610, 615, 720}; | ||
+ | float[] AMPLITUDES = {0, | ||
+ | ChannelSPD GREEN_SPD = new ChannelSPD(new IrregularSpectralCurve(WAVELENGTHS, | ||
+ | |||
+ | ChannelSPD RED_SPD = new ChannelSPD(new RGBSpectralCurve(0.8, | ||
+ | ChannelSPD CONST_SPD = new ChannelSPD(new ConstantSpectralCurve(0.25)); | ||
+ | |||
+ | Phong myShader = new Phong(); | ||
+ | myShader.setDiffuse(GREEN_SPD); | ||
+ | myShader.setTransparency(RED_SPD); | ||
+ | |||
+ | |||
+ | //within a module that extends a Box | ||
+ | module TestBox extends Box(0.001, | ||
+ | | ||
+ | //define a variable of type Phong and call it myShader | ||
+ | Phong myShader = new Phong(); | ||
+ | | ||
+ | //within the static constructor function - automatically called whenever a TestBox is generated | ||
+ | // the shader is parametrized and applied to the TestBox | ||
+ | { | ||
+ | //define the shader aspects | ||
+ | myShader.setDiffuse(GREEN_SPD); | ||
+ | myShader.setTransparency(RED_SPD); | ||
+ | myShader.setSpecular(CONST_SPD); | ||
+ | | ||
+ | //set the shader to the TestBox | ||
+ | setShader(myShader); | ||
+ | } | ||
+ | } | ||
+ | |||
+ | //as a module that is interpreted as Box | ||
+ | module TestBox ==> { | ||
+ | Phong myShader = new Phong(); | ||
+ | } Box(0.001, | ||
+ | </ | ||
+ | |||
+ | Within the // | ||
+ | |||
+ | * on 1nm resolution [300, 780] | ||
+ | * A, D65 | ||
+ | * on 5nm resolution [300, 780] | ||
+ | * A, C, D50, D55, D65, D75 | ||
+ | * on 5nm resolution [380, 780] | ||
+ | * FL1-12, FL3_1-15, HP1-5 | ||
+ | |||
+ | For instance, to use the predefined CIE NORM D65 for typical sun light within a user defined light module, one could use the following code. | ||
+ | |||
+ | <code java> | ||
+ | import de.grogra.imp3d.objects.Attributes; | ||
+ | import de.grogra.gpuflux.imp3d.spectral.CIENormSpectralCurve; | ||
+ | |||
+ | //define a light module | ||
+ | module MyLamp extends LightNode { | ||
+ | { | ||
+ | setLight( | ||
+ | new SpectralLight( | ||
+ | new CIENormSpectralCurve(Attributes.CIE_NORM_D65) | ||
+ | ).( | ||
+ | setPower(100), | ||
+ | setLight( | ||
+ | new SpotLight(DISTRIBUTION).( | ||
+ | setVisualize(true), | ||
+ | setNumberofrays(250), | ||
+ | setRaylength(1)// | ||
+ | ) | ||
+ | ) | ||
+ | ) //end SpectralLight | ||
+ | ); //end setLight | ||
+ | } | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | |||
+ | |||
+ | Note: Do NOT mix common RGB shader like the // | ||
+ | |||
+ | |||
+ | ==== Sensor nodes ==== | ||
+ | |||
+ | To monitor light distributions with a scene without interfering, | ||
+ | |||
+ | Note: The size of the sensor node directly correlates with the probability of got hit by a light ray. For a very small sphere the probability to got hit by a light ray is relatively low, so the number of light rays simulated by the light model needs to be much larger to get repayable results. Therefore, better not to use very small sensor nodes. | ||
+ | |||
+ | Note: The colour of the sensor node determines which wavelengths should be observed. The default value is white, what stands for monitor all colours. If, for instance, the sensor colour is set to red, only red spectra will be sensed. | ||
+ | |||
+ | Note: The output of a sensor node is normalized to absorbed radiance per square meter, independent of the actual size of the sensor. | ||
+ | |||
+ | Note: Sensor nodes can be enabled and disabled for the light model using the LM.setEnableSensors(true/ | ||
+ | |||
+ | <code java> | ||
+ | // create a 5cm, white sensor node | ||
+ | Axiom ==> SensorNode().(setRadius(0.05), | ||
+ | |||
+ | //check what the sensor node has sensed | ||
+ | x: | ||
+ | Measurement spectrum = lm.getSensedIrradianceMeasurement(x); | ||
+ | float absorbedPower = spectrum.integrate(); | ||
+ | ... | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | |||
+ | ==== SPD and PLD files and references ==== | ||
+ | Beside defining the SPD and PLD as arrays within XL, GroIMP supports the import of common file formates for both. | ||
+ | |||
+ | Both can be imported in the same way by open the Panels Tab and go to Explorers first. For PLD, the ' | ||
+ | |||
+ | {{ : | ||
+ | {{ : | ||
+ | |||
+ | To access the imported light spectra and physical light distributions within XL, one needs to define a reference to the files in the following way: | ||
+ | |||
+ | <code java> | ||
+ | // | ||
+ | const LightDistributionRef DISTRIBUTION = light(”distri1”); | ||
+ | const SpectrumRef SPECTRUM = spectrum(”equal”); | ||
+ | |||
+ | set them via the constructor | ||
+ | module MyLamp extends LightNode { | ||
+ | { | ||
+ | setLight(new SpectralLight(new PhysicalLight(DISTRIBUTION), | ||
+ | } | ||
+ | } | ||
+ | </ | ||
==== Example ==== | ==== Example ==== | ||
+ | In the following four minimal working examples are given to illustrate: the light model, the definition of light sources, adding a object and define a spectral shader, and on how to visualize the results. The four examples are building on each other, meaning with each example new parts will extend the previous code. | ||
+ | Note: The examples require GroIMP version >=2.0 to run. With GroIMP version 2.0 some changes on the internal package structure are made. formally classes found in // | ||
+ | |||
+ | <code java> | ||
+ | import de.grogra.imp3d.spectral.IrregularSpectralCurve; | ||
+ | import de.grogra.gpuflux.imp3d.spectral.IrregularSpectralCurve; | ||
+ | |||
+ | // Light nodes need to be imported like this | ||
+ | import de.grogra.gpuflux.imp3d.objects.PhysicalLight; | ||
+ | </ | ||
+ | |||
+ | === Example 1 - Light Model === | ||
+ | This example just defines the GPUFlux light model and parameterizes it to simulate a spectrum from 300 to 800nm and measure the results in 30 buckets. | ||
+ | |||
+ | <code java> | ||
+ | import de.grogra.gpuflux.tracer.FluxLightModelTracer.MeasureMode; | ||
+ | |||
+ | //constants for the light model: number of rays and maximal recursion depth | ||
+ | const int RAYS = 1000000; | ||
+ | const int DEPTH = 10; | ||
+ | |||
+ | // | ||
+ | protected void init () { | ||
+ | // | ||
+ | println(" | ||
+ | FluxLightModel GPU_LM = new FluxLightModel(RAYS, | ||
+ | GPU_LM.setSeed(12345); | ||
+ | GPU_LM.setMeasureMode(MeasureMode.FULL_SPECTRUM); | ||
+ | GPU_LM.setSpectralDomain(300, | ||
+ | GPU_LM.setSpectralBuckets(31);// | ||
+ | GPU_LM.compute();// | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | The model will run and directly when saved, creates an instance of the light model, set wanted parameters and run it. there will be no output (except the one form the light model itself, stating that it was executed and giving some statistics on the scene). | ||
+ | |||
+ | If you already get errors here, your system most probably does not support spectral light modelling. | ||
+ | |||
+ | |||
+ | === Example 2 - Light Sources === | ||
+ | This example defines a spectral light source with a user define physical light distribution (PLD) and a predefined CIE NORM D55 as spectral power distribution (SPD) (used to define typical sun light) and add the light source to the scene. | ||
+ | |||
+ | <code java> | ||
+ | import de.grogra.gpuflux.imp3d.spectral.CIENormSpectralCurve; | ||
+ | import de.grogra.gpuflux.imp3d.spectral.IrregularSpectralCurve; | ||
+ | import de.grogra.gpuflux.imp3d.objects.SpectralLight; | ||
+ | import de.grogra.gpuflux.imp3d.objects.PhysicalLight; | ||
+ | import de.grogra.gpuflux.tracer.FluxLightModelTracer.MeasureMode; | ||
+ | import de.grogra.gpuflux.scene.experiment.Measurement; | ||
+ | |||
+ | //////////////////////////////////////////////////////////////////////////////// | ||
+ | // | ||
+ | const double[][] DISTRIBUTION = { | ||
+ | {1, 1, 1,1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, | ||
+ | {1, 1, 1,1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, | ||
+ | {1, 1, 1,1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, | ||
+ | {1, 1, 1,1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, | ||
+ | {1, 1, 1,1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, | ||
+ | {1, 1, 1,1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, | ||
+ | {1, 1, 1,1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0} | ||
+ | }; | ||
+ | |||
+ | //use the predefined CIE NORM D65 for typical sun light | ||
+ | static const float[] WAVELENGTHS_S = CIENormSpectralCurve.NM_300_780_5; | ||
+ | static const float[] AMPLITUDES_S = CIENormSpectralCurve.D65; | ||
+ | |||
+ | //define a light node | ||
+ | module MyLamp extends LightNode { | ||
+ | { | ||
+ | setLight( | ||
+ | new SpectralLight( | ||
+ | new IrregularSpectralCurve(WAVELENGTHS_S, | ||
+ | ).( | ||
+ | setPower(7.8), | ||
+ | setLight( | ||
+ | new PhysicalLight(DISTRIBUTION).( | ||
+ | setVisualize(true), | ||
+ | setNumberofrays(500), | ||
+ | setRaylength(3.5) | ||
+ | ) | ||
+ | ) | ||
+ | ) //end SpectralLight | ||
+ | ); //end setLight | ||
+ | } | ||
+ | } | ||
+ | |||
+ | //constants for the light model: number of rays and maximal recursion depth | ||
+ | const int RAYS = 1000000; | ||
+ | const int DEPTH = 10; | ||
+ | |||
+ | // | ||
+ | protected void init () { | ||
+ | //create the actual 3D scene | ||
+ | [ | ||
+ | Axiom ==> MyLamp; | ||
+ | ] | ||
+ | |||
+ | //make sure the changes on the graph are applied... | ||
+ | {derive(); | ||
+ | //so that we directly can continue and work on the graph | ||
+ | |||
+ | // | ||
+ | println(" | ||
+ | FluxLightModel GPU_LM = new FluxLightModel(RAYS, | ||
+ | GPU_LM.setSeed(12345); | ||
+ | GPU_LM.setMeasureMode(MeasureMode.FULL_SPECTRUM); | ||
+ | GPU_LM.setSpectralDomain(300, | ||
+ | GPU_LM.setSpectralBuckets(31);// | ||
+ | GPU_LM.compute();// | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | Since the visualization of the light rays is turned on for the light source, we can see the light source in the 3D view window. | ||
+ | |||
+ | {{ : | ||
+ | |||
+ | |||
+ | === Example 3 - Scene object and shader === | ||
+ | Here now we define a test object, s simple flat box of one square meter in dimension and apply a green spectral shader to it. The 3D view window should now show something similar to this: | ||
+ | |||
+ | {{ : | ||
+ | |||
+ | <code java> | ||
+ | import de.grogra.gpuflux.imp3d.spectral.CIENormSpectralCurve; | ||
+ | import de.grogra.gpuflux.imp3d.shading.ChannelSPD; | ||
+ | import de.grogra.gpuflux.imp3d.spectral.IrregularSpectralCurve; | ||
+ | import de.grogra.gpuflux.imp3d.objects.SpectralLight; | ||
+ | import de.grogra.gpuflux.imp3d.objects.PhysicalLight; | ||
+ | import de.grogra.gpuflux.tracer.FluxLightModelTracer.MeasureMode; | ||
+ | import de.grogra.gpuflux.scene.experiment.Measurement; | ||
+ | |||
+ | //////////////////////////////////////////////////////////////////////////////// | ||
+ | // | ||
+ | const double[][] DISTRIBUTION = { | ||
+ | {1, 1, 1,1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, | ||
+ | {1, 1, 1,1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, | ||
+ | {1, 1, 1,1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, | ||
+ | {1, 1, 1,1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, | ||
+ | {1, 1, 1,1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, | ||
+ | {1, 1, 1,1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, | ||
+ | {1, 1, 1,1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0} | ||
+ | }; | ||
+ | |||
+ | //use the predefined CIE NORM D65 for typical sun light | ||
+ | static const float[] WAVELENGTHS_S = CIENormSpectralCurve.NM_300_780_5; | ||
+ | static const float[] AMPLITUDES_S = CIENormSpectralCurve.D65; | ||
+ | |||
+ | //define a light node | ||
+ | module MyLamp extends LightNode { | ||
+ | { | ||
+ | setLight( | ||
+ | new SpectralLight( | ||
+ | new IrregularSpectralCurve(WAVELENGTHS_S, | ||
+ | ).( | ||
+ | setPower(7.8), | ||
+ | setLight( | ||
+ | new PhysicalLight(DISTRIBUTION).( | ||
+ | setVisualize(true), | ||
+ | setNumberofrays(500), | ||
+ | setRaylength(3.5) | ||
+ | ) | ||
+ | ) | ||
+ | ) //end SpectralLight | ||
+ | ); //end setLight | ||
+ | |||
+ | } | ||
+ | } | ||
+ | |||
+ | //////////////////////////////////////////////////////////////////////////////// | ||
+ | //define a green shader as user-defined irregular spectral curve | ||
+ | public const float[] WAVELENGTHS = {300, 525, 530, 575, 580, 800}; | ||
+ | public const float[] AMPLITUDES = {0, | ||
+ | const ChannelSPD GREEN_SPD = new ChannelSPD(new IrregularSpectralCurve(WAVELENGTHS, | ||
+ | |||
+ | //apply the shader to an object: a box of one square meter | ||
+ | module TestShader ==> { | ||
+ | Phong myShader = new Phong(); | ||
+ | // | ||
+ | myShader.setDiffuse(GREEN_SPD); | ||
+ | } Box(0.001, | ||
+ | //////////////////////////////////////////////////////////////////////////////// | ||
+ | |||
+ | //constants for the light model: number of rays and maximal recursion depth | ||
+ | const int RAYS = 1000000; | ||
+ | const int DEPTH = 10; | ||
+ | |||
+ | // | ||
+ | protected void init () { | ||
+ | clearConsole(); | ||
+ | |||
+ | //create the actual 3D scene | ||
+ | [ | ||
+ | Axiom ==> TestShader M(2) RL(180) MyLamp; | ||
+ | ] | ||
+ | |||
+ | //make sure the changes on the graph are applied... | ||
+ | {derive(); | ||
+ | //so that we directly can continue and work on the graph | ||
+ | |||
+ | // | ||
+ | println(" | ||
+ | FluxLightModel GPU_LM = new FluxLightModel(RAYS, | ||
+ | GPU_LM.setSeed(12345); | ||
+ | GPU_LM.setMeasureMode(MeasureMode.FULL_SPECTRUM); | ||
+ | GPU_LM.setSpectralDomain(300, | ||
+ | GPU_LM.setSpectralBuckets(31);// | ||
+ | GPU_LM.compute();// | ||
+ | |||
+ | //check the scene objects for their light absorption | ||
+ | Measurement ms; | ||
+ | [ | ||
+ | x: | ||
+ | ] | ||
+ | println("" | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | To obtain the measurement results, one needs to first run the light model and second check each (wanted) scene object for its absorption values. A simple graph query can eb used to implement the second part, where here is searched for all TestShader instances within the graph and the light absorption is obtained. Afterwards the results a printed to the GroIMP console window. The output of the code is the integrated absorbed power and the array of the absorption values for each bucket. | ||
+ | |||
+ | {{ : | ||
+ | |||
+ | |||
+ | === Example 4 - Output visualization === | ||
+ | In the final version, we are now going to add a charts to visualize the emitted spectrum and to plot it against the absorbed spectrum of the test object. | ||
+ | |||
+ | {{ : | ||
+ | |||
+ | <code java> | ||
+ | import de.grogra.gpuflux.imp3d.spectral.CIENormSpectralCurve; | ||
+ | import de.grogra.gpuflux.imp3d.shading.ChannelSPD; | ||
+ | import de.grogra.gpuflux.imp3d.spectral.IrregularSpectralCurve; | ||
+ | import de.grogra.gpuflux.imp3d.objects.SpectralLight; | ||
+ | import de.grogra.gpuflux.imp3d.objects.PhysicalLight; | ||
+ | import de.grogra.gpuflux.tracer.FluxLightModelTracer.MeasureMode; | ||
+ | import de.grogra.gpuflux.scene.experiment.Measurement; | ||
+ | |||
+ | //////////////////////////////////////////////////////////////////////////////// | ||
+ | // | ||
+ | const double[][] DISTRIBUTION = { | ||
+ | {1, 1, 1,1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, | ||
+ | {1, 1, 1,1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, | ||
+ | {1, 1, 1,1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, | ||
+ | {1, 1, 1,1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, | ||
+ | {1, 1, 1,1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, | ||
+ | {1, 1, 1,1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}, | ||
+ | {1, 1, 1,1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0} | ||
+ | }; | ||
+ | |||
+ | |||
+ | //use the predefined CIE NORM D65 for typical sun light | ||
+ | static const float[] WAVELENGTHS_S = CIENormSpectralCurve.NM_300_780_5; | ||
+ | static const float[] AMPLITUDES_S = CIENormSpectralCurve.D65; | ||
+ | |||
+ | //define a light node | ||
+ | module MyLamp extends LightNode { | ||
+ | { | ||
+ | setLight( | ||
+ | new SpectralLight( | ||
+ | new IrregularSpectralCurve(WAVELENGTHS_S, | ||
+ | ).( | ||
+ | setPower(7.8), | ||
+ | setLight( | ||
+ | new PhysicalLight(DISTRIBUTION).( | ||
+ | setVisualize(true), | ||
+ | setNumberofrays(500), | ||
+ | setRaylength(3.5) | ||
+ | ) | ||
+ | ) | ||
+ | ) //end SpectralLight | ||
+ | ); //end setLight | ||
+ | |||
+ | } | ||
+ | } | ||
+ | |||
+ | //////////////////////////////////////////////////////////////////////////////// | ||
+ | //define a green shader as user-defined irregular spectral curve | ||
+ | public const float[] WAVELENGTHS = {300, 525, 530, 575, 580, 700}; | ||
+ | public const float[] AMPLITUDES = {0, | ||
+ | const ChannelSPD GREEN_SPD = new ChannelSPD(new IrregularSpectralCurve(WAVELENGTHS, | ||
+ | |||
+ | //apply the shader to an object: a box of one square meter | ||
+ | module TestShader ==> { | ||
+ | Phong myShader = new Phong(); | ||
+ | // | ||
+ | myShader.setDiffuse(GREEN_SPD); | ||
+ | } Box(0.001, | ||
+ | //////////////////////////////////////////////////////////////////////////////// | ||
+ | |||
+ | //define the data sheet | ||
+ | const DatasetRef absorbedChart = new DatasetRef(" | ||
+ | |||
+ | //constants for the light model: number of rays and maximal recursion depth | ||
+ | const int RAYS = 1000000; | ||
+ | const int DEPTH = 10; | ||
+ | |||
+ | // | ||
+ | protected void init () { | ||
+ | clearConsole(); | ||
+ | |||
+ | // | ||
+ | absorbedChart.clear().setColumnKey(0," | ||
+ | chart(absorbedChart, | ||
+ | |||
+ | //plot the emitted spectral curve | ||
+ | float INTEGRAL = 0; | ||
+ | for(int i: | ||
+ | for(int i: | ||
+ | absorbedChart.addRow().(set(0, | ||
+ | } | ||
+ | |||
+ | //create the actual 3D scene | ||
+ | [ | ||
+ | Axiom ==> TestShader M(2) RL(180) MyLamp; | ||
+ | ] | ||
+ | |||
+ | //make sure the changes on the graph are applied... | ||
+ | {derive(); | ||
+ | //so that we directly can continue and work on the graph | ||
+ | |||
+ | // | ||
+ | println(" | ||
+ | FluxLightModel GPU_LM = new FluxLightModel(RAYS, | ||
+ | GPU_LM.setSeed(12345); | ||
+ | GPU_LM.setMeasureMode(MeasureMode.FULL_SPECTRUM); | ||
+ | GPU_LM.setSpectralDomain(300, | ||
+ | GPU_LM.setSpectralBuckets(31);// | ||
+ | GPU_LM.compute();// | ||
+ | |||
+ | //check the scene objects for their light absorption | ||
+ | Measurement ms; | ||
+ | [ | ||
+ | x: | ||
+ | ] | ||
+ | print(" | ||
+ | |||
+ | //plot the absorption spectrum | ||
+ | for(int i: | ||
+ | absorbedChart.addRow().(set(1, | ||
+ | } | ||
+ | } | ||
+ | </ | ||
Line 299: | Line 744: | ||
Special thanks to Dietger van Antwerpen, who implemented the GPUFlux light model for GroIMP! | Special thanks to Dietger van Antwerpen, who implemented the GPUFlux light model for GroIMP! | ||
+ | |||
+ | |||
+ | ===== References ===== | ||
+ | |||
+ | * Henke M and Buck-Sorlin GH (2018) Using a full spectral raytracer for the modelling of light microclimate in a functional-structural plant model; Computing and Informatics, | ||
+ | * van Antwerpen, D.G. (2011) High Performance Spectral Light Transport Model for Agricultural Applications, | ||
+ | * van Antwerpen, D.G. (2011) Unbiased physically based rendering on the GPU, Master thesis, Delft University of Technology | ||
+ |
tutorials/basic-spectral-light-modeling.1733129874.txt.gz · Last modified: 2024/12/02 09:57 by MH