diff --git a/README.md b/README.md
index bf267c46e18f252f28cb66a788c08af445ebac11..c1869f1f1d92d035fac176f475e81d0e7af5f2e2 100644
--- a/README.md
+++ b/README.md
@@ -351,6 +351,47 @@ The IFOs included in `gwinc.ifo` provide examples of the use of the
 budget interface (e.g. [gwinc.ifo.aLIGO](gwinc/ifo/aLIGO)).
 
 
+### "precomp" functions
+
+`BudgetItems` support "precomp" functions that are calculated during
+the `update()` method call and can be used to cache common derived
+values in the `ifo` struct for later use.  The execution state of
+these functions is cached at the Budget level, to prevent needlessly
+re-calculating them in multiple BudgetItems.  When they are calculated
+they are provided with the `freq` and `ifo` attributes as arguments.
+For example, take the following budget definition:
+```python
+
+class Noise0(Noise):
+    precomp = [
+        precomp_foo,
+    ]
+    ...
+
+class Noise1(Noise):
+    precomp = [
+        precomp_foo,
+        precomp_bar,
+    ]
+    ...
+
+class MyBudget(Budget):
+    noises = [
+        Noise0,
+        Noise1,
+    ]
+```
+When the `MyBudget.update()` method is called, all the `precomp`
+functions will be execute:
+```python
+precomp_foo(self.freq, self.ifo)
+precomp_bar(self.freq, self.ifo)
+```
+But the `precomp_foo` function will only be calculated once per budget
+update() call, even though it's specified as needed by both `Noise0`
+and `Noise1`.
+
+
 ### extracting single noise terms
 
 There are various way to extract single noise terms from the Budget
diff --git a/gwinc/ifo/__main__.py b/gwinc/ifo/__main__.py
index 84f5f1b08ad11303ad7d5ec278757c863394f251..1da5c7c0a65ce61ce665c1d3af6b6d40324c01fd 100644
--- a/gwinc/ifo/__main__.py
+++ b/gwinc/ifo/__main__.py
@@ -35,6 +35,7 @@ def main():
         range_pad = max(len(name), range_pad)
 
     for name, budget in budgets.items():
+        budget.update()
         data = budget.calc()
         try:
             import inspiral_range
diff --git a/gwinc/ifo/noises.py b/gwinc/ifo/noises.py
index 06770cf4b3aba292d9872717078f286903fc27ea..88c701559312aa325bd249825e605da9c88a1dcd 100644
--- a/gwinc/ifo/noises.py
+++ b/gwinc/ifo/noises.py
@@ -127,15 +127,6 @@ def precomp_mirror(f, ifo):
         * ifo.Materials.Substrate.MassDensity
 
 
-def precomp_gwinc(f, ifo):
-    ifo.gwinc = Struct()
-    pbs, parm, finesse, prfactor, Tpr = ifo_power(ifo)
-    ifo.gwinc.pbs = pbs
-    ifo.gwinc.parm = parm
-    ifo.gwinc.finesse = finesse
-    ifo.gwinc.prfactor = prfactor
-
-
 def precomp_suspension(f, ifo):
     if 'VHCoupling' not in ifo.Suspension:
         ifo.Suspension.VHCoupling = Struct()
@@ -146,6 +137,31 @@ def precomp_suspension(f, ifo):
     ifo.Suspension.hTable = hTable
     ifo.Suspension.vTable = vTable
 
+
+def precomp_coating(f, ifo):
+    ifo.Optics.ITM.dOpt = coating_thickness(ifo, 'ITM')
+    ifo.Optics.ETM.dOpt = coating_thickness(ifo, 'ETM')
+
+
+def precomp_power(f, ifo):
+    if 'gwinc' not in ifo:
+        ifo.gwinc = Struct()
+    pbs, parm, finesse, prfactor, Tpr = ifo_power(ifo)
+    ifo.gwinc.pbs = pbs
+    ifo.gwinc.parm = parm
+    ifo.gwinc.finesse = finesse
+    ifo.gwinc.gPhase = finesse * 2/np.pi
+    ifo.gwinc.prfactor = prfactor
+
+
+def precomp_cavity(f, ifo):
+    if 'gwinc' not in ifo:
+        ifo.gwinc = Struct()
+    w0, wBeam_ITM, wBeam_ETM = arm_cavity(ifo)
+    ifo.gwinc.w0 = w0
+    ifo.gwinc.wBeam_ITM = wBeam_ITM
+    ifo.gwinc.wBeam_ETM = wBeam_ETM
+
 ##################################################
 
 class Strain(nb.Calibration):
@@ -164,9 +180,12 @@ class QuantumVacuum(nb.Noise):
         color='#ad03de',
     )
 
+    precomp = [
+        precomp_mirror,
+        precomp_power,
+    ]
+
     def calc(self):
-        precomp_mirror(self.freq, self.ifo)
-        precomp_gwinc(self.freq, self.ifo)
         return noise.quantum.shotrad(self.freq, self.ifo)
 
 
@@ -193,8 +212,11 @@ class Seismic(nb.Noise):
         color='#855700',
     )
 
+    precomp = [
+        precomp_suspension,
+    ]
+
     def calc(self):
-        precomp_suspension(self.freq, self.ifo)
         if 'PlatformMotion' in self.ifo.Seismic:
             if self.ifo.Seismic.PlatformMotion == 'BSC':
                 nt, nr = noise.seismic.seismic_BSC_ISI(self.freq)
@@ -275,8 +297,11 @@ class SuspensionThermal(nb.Noise):
         color='#0d75f8',
     )
 
+    precomp = [
+        precomp_suspension,
+    ]
+
     def calc(self):
-        precomp_suspension(self.freq, self.ifo)
         n = noise.suspensionthermal.suspension_thermal(self.freq, self.ifo.Suspension)
         return n * 4
 
@@ -290,16 +315,20 @@ class CoatingBrownian(nb.Noise):
         color='#fe0002',
     )
 
+    precomp = [
+        precomp_cavity,
+        precomp_coating,
+    ]
+
     def calc(self):
         wavelength = self.ifo.Laser.Wavelength
         materials = self.ifo.Materials
-        w0, wBeam_ITM, wBeam_ETM = arm_cavity(self.ifo)
-        dOpt_ITM = coating_thickness(self.ifo, 'ITM')
-        dOpt_ETM = coating_thickness(self.ifo, 'ETM')
         nITM = noise.coatingthermal.coating_brownian(
-            self.freq, materials, wavelength, wBeam_ITM, dOpt_ITM)
+            self.freq, materials, wavelength,
+            self.ifo.gwinc.wBeam_ITM, self.ifo.Optics.ITM.dOpt)
         nETM = noise.coatingthermal.coating_brownian(
-            self.freq, materials, wavelength, wBeam_ETM, dOpt_ETM)
+            self.freq, materials, wavelength,
+            self.ifo.gwinc.wBeam_ETM, self.ifo.Optics.ETM.dOpt)
         return (nITM + nETM) * 2
 
 
@@ -313,16 +342,20 @@ class CoatingThermoOptic(nb.Noise):
         linestyle='--',
     )
 
+    precomp = [
+        precomp_cavity,
+        precomp_coating,
+    ]
+
     def calc(self):
         wavelength = self.ifo.Laser.Wavelength
         materials = self.ifo.Materials
-        w0, wBeam_ITM, wBeam_ETM = arm_cavity(self.ifo)
-        dOpt_ITM = coating_thickness(self.ifo, 'ITM')
-        dOpt_ETM = coating_thickness(self.ifo, 'ETM')
         nITM, junk1, junk2, junk3 = noise.coatingthermal.coating_thermooptic(
-            self.freq, materials, wavelength, wBeam_ITM, dOpt_ITM[:])
+            self.freq, materials, wavelength,
+            self.ifo.gwinc.wBeam_ITM, self.ifo.Optics.ITM.dOpt[:])
         nETM, junk1, junk2, junk3 = noise.coatingthermal.coating_thermooptic(
-            self.freq, materials, wavelength, wBeam_ETM, dOpt_ETM[:])
+            self.freq, materials, wavelength,
+            self.ifo.gwinc.wBeam_ETM, self.ifo.Optics.ETM.dOpt[:])
         return (nITM + nETM) * 2
 
 
@@ -336,13 +369,15 @@ class ITMThermoRefractive(nb.Noise):
         linestyle='--',
     )
 
+    precomp = [
+        precomp_power,
+        precomp_cavity,
+    ]
+
     def calc(self):
-        finesse = ifo_power(self.ifo)[2]
-        gPhase = finesse * 2/np.pi
-        w0, wBeam_ITM, wBeam_ETM = arm_cavity(self.ifo)
         n = noise.substratethermal.substrate_thermorefractive(
-            self.freq, self.ifo.Materials, wBeam_ITM)
-        return n * 2 / gPhase**2
+            self.freq, self.ifo.Materials, self.ifo.gwinc.wBeam_ITM)
+        return n * 2 / self.ifo.gwinc.gPhase**2
 
 
 class ITMCarrierDensity(nb.Noise):
@@ -355,13 +390,15 @@ class ITMCarrierDensity(nb.Noise):
         linestyle='--',
     )
 
+    precomp = [
+        precomp_power,
+        precomp_cavity,
+    ]
+
     def calc(self):
-        finesse = ifo_power(self.ifo)[2]
-        gPhase = finesse * 2/np.pi
-        w0, wBeam_ITM, wBeam_ETM = arm_cavity(self.ifo)
         n = noise.substratethermal.substrate_carrierdensity(
-            self.freq, self.ifo.Materials, wBeam_ITM)
-        return n * 2 / gPhase**2
+            self.freq, self.ifo.Materials, self.ifo.gwinc.wBeam_ITM)
+        return n * 2 / self.ifo.gwinc.gPhase**2
 
 
 class SubstrateBrownian(nb.Noise):
@@ -374,12 +411,15 @@ class SubstrateBrownian(nb.Noise):
         linestyle='--',
     )
 
+    precomp = [
+        precomp_cavity,
+    ]
+
     def calc(self):
-        w0, wBeam_ITM, wBeam_ETM = arm_cavity(self.ifo)
         nITM = noise.substratethermal.substrate_brownian(
-            self.freq, self.ifo.Materials, wBeam_ITM)
+            self.freq, self.ifo.Materials, self.ifo.gwinc.wBeam_ITM)
         nETM = noise.substratethermal.substrate_brownian(
-            self.freq, self.ifo.Materials, wBeam_ETM)
+            self.freq, self.ifo.Materials, self.ifo.gwinc.wBeam_ETM)
         return (nITM + nETM) * 2
 
 
@@ -393,12 +433,15 @@ class SubstrateThermoElastic(nb.Noise):
         linestyle='--',
     )
 
+    precomp = [
+        precomp_cavity,
+    ]
+
     def calc(self):
-        w0, wBeam_ITM, wBeam_ETM = arm_cavity(self.ifo)
         nITM = noise.substratethermal.substrate_thermoelastic(
-            self.freq, self.ifo.Materials, wBeam_ITM)
+            self.freq, self.ifo.Materials, self.ifo.gwinc.wBeam_ITM)
         nETM = noise.substratethermal.substrate_thermoelastic(
-            self.freq, self.ifo.Materials, wBeam_ETM)
+            self.freq, self.ifo.Materials, self.ifo.gwinc.wBeam_ETM)
         return (nITM + nETM) * 2
 
 
diff --git a/gwinc/nb.py b/gwinc/nb.py
index 437baea5b52d911730f86694d7c21edbfb47c4e4..2680862aceeb7c2e751672a0f352674c0d14e749 100644
--- a/gwinc/nb.py
+++ b/gwinc/nb.py
@@ -31,15 +31,44 @@ class BudgetItem:
         """
         return None
 
-    def update(self, **kwargs):
-        """Overload method for updating data.
+    def update(self, _precomp=None, **kwargs):
+        """Update parameters and execute precomp functions.
 
-        By default any keyword arguments provided are written directly
-        as attribute variables (as with __init__).
+        The method does two things.  First, any keyword arguments
+        provided are written directly as attribute variables to the
+        class (as with __init__).
+
+        Second, after attribute update, all functions listed in the
+        `precomp` attribute are executed, supplied with the `freq` and
+        `ifo` attributes.  So if the `precomp` attribute is defined
+        as:
+
+          precomp = [precomp_foo, precomp_bar]
+
+        then this method will execute the following after attribute
+        update:
+
+          precomp_foo(self.freq, self.ifo)
+          precomp_bar(self.freq, self.ifo)
+
+        These functions are intended to update the `ifo` Struct
+        attribute with derived values cached for later usage.  If the
+        `_precomp` argument is provided to this method then it is
+        assumed to be a set of previously executed precomp functions,
+        and any function already listed there will not be re-executed,
+        and it will be updated with any newly executed functions.
 
         """
         for key, val in kwargs.items():
             setattr(self, key, val)
+        for func in getattr(self, 'precomp', []):
+            if _precomp is None:
+                _precomp = set()
+            if func in _precomp:
+                continue
+            logger.debug("precomp {}".format(func))
+            func(self.freq, self.ifo)
+            _precomp.add(func)
 
     def calc(self):
         """Overload method for final PSD calculation.
@@ -361,13 +390,15 @@ class Budget(Noise):
             logger.debug("load {}".format(item))
             item.load()
 
-    def update(self, **kwargs):
+    def update(self, _precomp=None, **kwargs):
         """Update all noise and cal objects with supplied kwargs."""
+        if _precomp is None:
+            _precomp = set()
         for name, item in itertools.chain(
                 self._cal_objs.items(),
                 self._noise_objs.items()):
             logger.debug("update {}".format(item))
-            item.update(**kwargs)
+            item.update(_precomp=_precomp, **kwargs)
 
     def calc_noise(self, name):
         """Return calibrated individual noise term.