diff --git a/vp8/common/blockd.h b/vp8/common/blockd.h
index 690e0f8c8492921564a1ad80f9ce04fd8c5c1e15..c477d2d6d9fb74f166d82a1fdbe076a639c14859 100644
--- a/vp8/common/blockd.h
+++ b/vp8/common/blockd.h
@@ -32,6 +32,7 @@ void vpx_log(const char *format, ...);
 #define DCPREDCNTTHRESH 3
 
 #define MB_FEATURE_TREE_PROBS   3
+#define SEGMENT_PREDICTION_PROBS 3
 
 #define MAX_MB_SEGMENTS         4
 
@@ -187,6 +188,7 @@ typedef struct
     unsigned char mb_skip_coeff;                                /* does this mb has coefficients at all, 1=no coefficients, 0=need decode tokens */
     unsigned char need_to_clamp_mvs;
     unsigned char segment_id;                  /* Which set of segmentation parameters should be used for this MB */
+
 } MB_MODE_INFO;
 
 typedef struct
@@ -258,11 +260,14 @@ typedef struct MacroBlockD
 
     /* Per frame flags that define which MB level features (such as quantizer or loop filter level) */
     /* are enabled and when enabled the proabilities used to decode the per MB flags in MB_MODE_INFO */
+
+    // Probability Tree used to code Segment number
+    vp8_prob mb_segment_tree_probs[MB_FEATURE_TREE_PROBS];
+
 #if CONFIG_SEGMENTATION
-    vp8_prob mb_segment_tree_probs[MB_FEATURE_TREE_PROBS + 3];         // Probability Tree used to code Segment number
+    // Context probabilities when using predictive coding of segment id
+    vp8_prob mb_segment_pred_probs[SEGMENT_PREDICTION_PROBS];
     unsigned char temporal_update;
-#else
-    vp8_prob mb_segment_tree_probs[MB_FEATURE_TREE_PROBS];
 #endif
 
     // Segment features
diff --git a/vp8/decoder/decodemv.c b/vp8/decoder/decodemv.c
index e13fd6f09a642ce5faa838f0ff6d6ac266a761cb..416865cae5d34639109a8fba1073d657cbda1696 100644
--- a/vp8/decoder/decodemv.c
+++ b/vp8/decoder/decodemv.c
@@ -18,10 +18,6 @@
 //#if CONFIG_SEGFEATURES
 #include "vp8/common/seg_common.h"
 
-#if CONFIG_SEGMENTATION
-#include "vp8/common/seg_common.h"
-#endif
-
 #if CONFIG_DEBUG
 #include <assert.h>
 #endif
@@ -434,7 +430,7 @@ static void read_mb_modes_mv(VP8D_COMP *pbi, MODE_INFO *mi, MB_MODE_INFO *mbmi,
     MACROBLOCKD *const xd  = & pbi->mb;
 
 #if CONFIG_SEGMENTATION
-    int sum;
+    int pred_context;
     int index = mb_row * pbi->common.mb_cols + mb_col;
 #endif
     int_mv *const mv = & mbmi->mv;
@@ -465,14 +461,16 @@ static void read_mb_modes_mv(VP8D_COMP *pbi, MODE_INFO *mi, MB_MODE_INFO *mbmi,
 #if CONFIG_SEGMENTATION
                 if (xd->temporal_update)
                 {
-                    sum = 0;
+                    pred_context = 0;
 
                     if (mb_col != 0)
-                        sum += (mi-1)->mbmi.segment_flag;
+                        pred_context += (mi-1)->mbmi.segment_flag;
                     if (mb_row != 0)
-                        sum += (mi-pbi->common.mb_cols)->mbmi.segment_flag;
+                        pred_context +=
+                            (mi-pbi->common.mb_cols)->mbmi.segment_flag;
 
-                    if (vp8_read(bc, xd->mb_segment_tree_probs[3+sum]) == 0)
+                    if (vp8_read(bc,
+                                 xd->mb_segment_pred_probs[pred_context]) == 0)
                     {
                         mbmi->segment_id = pbi->segmentation_map[index];
                         mbmi->segment_flag = 0;
diff --git a/vp8/decoder/decodframe.c b/vp8/decoder/decodframe.c
index 5ba49915515a4d2ff8167bf627382b6528333b9e..c572b07149226882dc698579de7a5cba372d9237 100644
--- a/vp8/decoder/decodframe.c
+++ b/vp8/decoder/decodframe.c
@@ -1061,18 +1061,38 @@ int vp8_decode_frame(VP8D_COMP *pbi)
         if (xd->update_mb_segmentation_map)
         {
             /* Which macro block level features are enabled */
-            vpx_memset(xd->mb_segment_tree_probs, 255, sizeof(xd->mb_segment_tree_probs));
-#if CONFIG_SEGMENTATION
-            /* Read the probs used to decode the segment id for each macro block. */
-            for (i = 0; i < MB_FEATURE_TREE_PROBS+3; i++)
-#else
+            vpx_memset(xd->mb_segment_tree_probs, 255,
+                       sizeof(xd->mb_segment_tree_probs));
+            vpx_memset(xd->mb_segment_pred_probs, 255,
+                       sizeof(xd->mb_segment_pred_probs));
+
+            // Read the probs used to decode the segment id for each macro
+            // block.
             for (i = 0; i < MB_FEATURE_TREE_PROBS; i++)
-#endif
             {
-                /* If not explicitly set value is defaulted to 255 by memset above */
+                // If not explicitly set value is defaulted to 255 by
+                //memset above
                 if (vp8_read_bit(bc))
-                    xd->mb_segment_tree_probs[i] = (vp8_prob)vp8_read_literal(bc, 8);
+                    xd->mb_segment_tree_probs[i] =
+                        (vp8_prob)vp8_read_literal(bc, 8);
             }
+#if CONFIG_SEGMENTATION
+            // If predictive coding of segment map is enabled read the
+            // prediction probabilities.
+            if ( xd->temporal_update )
+            {
+                // Read the prediction probs needed to decode the segment id
+                // when predictive coding enabled
+                for (i = 0; i < SEGMENT_PREDICTION_PROBS; i++)
+                {
+                    // If not explicitly set value is defaulted to 255 by
+                    // memset above
+                    if (vp8_read_bit(bc))
+                        xd->mb_segment_pred_probs[i] =
+                            (vp8_prob)vp8_read_literal(bc, 8);
+                }
+            }
+#endif
         }
     }
 
diff --git a/vp8/encoder/bitstream.c b/vp8/encoder/bitstream.c
index ff8fb8a4816c9be2f13a46c04cd692b8e0328937..eda40f375e1a8bae6342a61faee1c8512dbb815e 100644
--- a/vp8/encoder/bitstream.c
+++ b/vp8/encoder/bitstream.c
@@ -64,10 +64,6 @@ extern unsigned int active_section;
 
 #ifdef MODE_STATS
 int count_mb_seg[4] = { 0, 0, 0, 0 };
-#if CONFIG_SEGMENTATION
-int segment_modes_intra[MAX_MB_SEGMENTS] = { 0, 0, 0, 0 };
-int segment_modes_inter[MAX_MB_SEGMENTS] = { 0, 0, 0, 0 };
-#endif
 #endif
 
 
@@ -945,7 +941,7 @@ static void pack_inter_mode_mvs(VP8_COMP *const cpi)
     MACROBLOCKD *xd = &cpi->mb.e_mbd;
 #if CONFIG_SEGMENTATION
     int i;
-    int sum;
+    int pred_context;
     int index = 0;
 #endif
     const int *const rfct = cpi->count_mb_ref_frame_usage;
@@ -1042,37 +1038,30 @@ static void pack_inter_mode_mvs(VP8_COMP *const cpi)
             active_section = 9;
 #endif
 
-#ifdef MODE_STATS
-#if CONFIG_SEGMENTATION
-            segment_modes_inter[segment_id]++;
-#endif
-#endif
             if (cpi->mb.e_mbd.update_mb_segmentation_map)
             {
 #if CONFIG_SEGMENTATION
                 if (xd->temporal_update)
                 {
-                    sum = 0;
+                    pred_context = 0;
                     if (mb_col != 0)
-                        sum +=  (m-1)->mbmi.segment_flag;
+                        pred_context +=  (m-1)->mbmi.segment_flag;
                     if (mb_row != 0)
-                        sum += (m-pc->mb_cols)->mbmi.segment_flag;
+                        pred_context += (m-pc->mb_cols)->mbmi.segment_flag;
 
                     if (m->mbmi.segment_flag == 0)
                     {
-                        vp8_write(w,0,xd->mb_segment_tree_probs[3+sum]);
+                        vp8_write(w,0,xd->mb_segment_pred_probs[pred_context]);
                     }
                     else
                     {
-                        vp8_write(w,1,xd->mb_segment_tree_probs[3+sum]);
+                        vp8_write(w,1,xd->mb_segment_pred_probs[pred_context]);
                         write_mb_segid(w, mi, &cpi->mb.e_mbd);
-                        cpi->segmentation_map[index] = segment_id;
                     }
                 }
                 else
                 {
                     write_mb_segid(w, mi, &cpi->mb.e_mbd);
-                    cpi->segmentation_map[index] = segment_id;
                 }
                 index++;
 #else
@@ -1268,21 +1257,12 @@ static void write_kfmodes(VP8_COMP *cpi)
             const int ym = m->mbmi.mode;
             int segment_id = m->mbmi.segment_id;
 
-#ifdef MODE_STATS
-#if CONFIG_SEGMENTATION
-            segment_modes_intra[segment_id]++;
-#endif
-#endif
-
             if (cpi->mb.e_mbd.update_mb_segmentation_map)
             {
 #if CONFIG_SEGMENTATION
-                write_mb_segid(bc, &m->mbmi, &cpi->mb.e_mbd);
-                cpi->segmentation_map[index] = segment_id;
                 index++;
-#else
-                write_mb_segid(bc, &m->mbmi, &cpi->mb.e_mbd);
 #endif
+                write_mb_segid(bc, &m->mbmi, &cpi->mb.e_mbd);
             }
 
 //#if CONFIG_SEGFEATURES
@@ -2050,12 +2030,9 @@ void vp8_pack_bitstream(VP8_COMP *cpi, unsigned char *dest, unsigned long *size)
 
         if (xd->update_mb_segmentation_map)
         {
- #if CONFIG_SEGMENTATION
-            // Write the probs used to decode the segment id for each macro block.
-            for (i = 0; i < MB_FEATURE_TREE_PROBS+3; i++)
-#else
+            // Send the tree probabilities used to decode unpredicted
+            // macro-block segments
             for (i = 0; i < MB_FEATURE_TREE_PROBS; i++)
-#endif
             {
                 int Data = xd->mb_segment_tree_probs[i];
 
@@ -2067,6 +2044,25 @@ void vp8_pack_bitstream(VP8_COMP *cpi, unsigned char *dest, unsigned long *size)
                 else
                     vp8_write_bit(bc, 0);
             }
+#if CONFIG_SEGMENTATION
+            // If predictive coding of segment map is enabled send the
+            // prediction probabilities.
+            if ( xd->temporal_update )
+            {
+                for (i = 0; i < SEGMENT_PREDICTION_PROBS; i++)
+                {
+                    int Data = xd->mb_segment_pred_probs[i];
+
+                    if (Data != 255)
+                    {
+                        vp8_write_bit(bc, 1);
+                        vp8_write_literal(bc, Data, 8);
+                    }
+                    else
+                        vp8_write_bit(bc, 0);
+                }
+            }
+#endif
         }
     }
 
diff --git a/vp8/encoder/encodeframe.c b/vp8/encoder/encodeframe.c
index b48a5f687a6a31fc4266b6ee54d8700080ebadf1..4cde41df430a52b2c4a691cca1fc681395d1bde1 100644
--- a/vp8/encoder/encodeframe.c
+++ b/vp8/encoder/encodeframe.c
@@ -578,9 +578,7 @@ void encode_mb_row(VP8_COMP *cpi,
     int recon_y_stride = cm->yv12_fb[ref_fb_idx].y_stride;
     int recon_uv_stride = cm->yv12_fb[ref_fb_idx].uv_stride;
     int map_index = (mb_row * cpi->common.mb_cols);
-#if CONFIG_SEGMENTATION
-    int sum;
-#endif
+
 #if CONFIG_MULTITHREAD
     const int nsync = cpi->mt_sync_range;
     const int rightmost_col = cm->mb_cols - 1;
@@ -768,43 +766,7 @@ void encode_mb_row(VP8_COMP *cpi,
         recon_yoffset += 16;
         recon_uvoffset += 8;
 
-#if CONFIG_SEGMENTATION
-       //cpi->segmentation_map[mb_row * cm->mb_cols + mb_col] =  xd->mbmi.segment_id;
-        if (cm->frame_type == KEY_FRAME)
-        {
-            segment_counts[xd->mode_info_context->mbmi.segment_id]++;
-        }
-        else
-        {
-            sum = 0;
-            if (mb_col != 0)
-                sum += (xd->mode_info_context-1)->mbmi.segment_flag;
-            if (mb_row != 0)
-                sum += (xd->mode_info_context-cm->mb_cols)->mbmi.segment_flag;
-
-            if ( xd->mode_info_context->mbmi.segment_id ==
-                 cpi->last_segmentation_map[(mb_row*cm->mb_cols) + mb_col] )
-            {
-                xd->mode_info_context->mbmi.segment_flag = 0;
-            }
-            else
-                xd->mode_info_context->mbmi.segment_flag = 1;
-
-            if (xd->mode_info_context->mbmi.segment_flag == 0)
-            {
-                segment_counts[SEEK_SAMEID + sum]++;
-                segment_counts[10]++;
-            }
-            else
-            {
-                segment_counts[SEEK_DIFFID + sum]++;
-                segment_counts[11]++;
-                //calculate individual segment ids
-                segment_counts[xd->mode_info_context->mbmi.segment_id] ++;
-            }
-        }
-        segment_counts[SEEK_SEGID + xd->mode_info_context->mbmi.segment_id] ++;
-#else
+#if !CONFIG_SEGMENTATION
         segment_counts[xd->mode_info_context->mbmi.segment_id] ++;
 #endif
         // skip to next mb
@@ -948,12 +910,7 @@ void vp8_encode_frame(VP8_COMP *cpi)
     MACROBLOCKD *const xd = & x->e_mbd;
 
     TOKENEXTRA *tp = cpi->tok;
-
-#if CONFIG_SEGMENTATION
-    int segment_counts[MAX_MB_SEGMENTS + SEEK_SEGID];
-#else
     int segment_counts[MAX_MB_SEGMENTS];
-#endif
     int totalrate;
 
 
@@ -1144,7 +1101,7 @@ void vp8_encode_frame(VP8_COMP *cpi)
 
 #if CONFIG_SEGMENTATION
         // Select the coding strategy for the segment map (temporal or spatial)
-        choose_segmap_coding_method( cpi, segment_counts );
+        choose_segmap_coding_method( cpi );
 #else
         tot_count = segment_counts[0] + segment_counts[1] + segment_counts[2] + segment_counts[3];
         count1 = segment_counts[0] + segment_counts[1];
@@ -1159,17 +1116,14 @@ void vp8_encode_frame(VP8_COMP *cpi)
         if (count2 > 0)
             xd->mb_segment_tree_probs[2] = (segment_counts[2] * 255) /count2;
 
-#endif
         // Zero probabilities not allowed
-#if CONFIG_SEGMENTATION
-            for (i = 0; i < MB_FEATURE_TREE_PROBS+3; i++)
-#else
-            for (i = 0; i < MB_FEATURE_TREE_PROBS; i++)
+        for (i = 0; i < MB_FEATURE_TREE_PROBS; i++)
+
+        {
+            if (xd->mb_segment_tree_probs[i] == 0)
+                xd->mb_segment_tree_probs[i] = 1;
+        }
 #endif
-            {
-                if (xd->mb_segment_tree_probs[i] == 0)
-                    xd->mb_segment_tree_probs[i] = 1;
-            }
     }
 
     // 256 rate units to the bit
diff --git a/vp8/encoder/onyx_if.c b/vp8/encoder/onyx_if.c
index 8dbb0059b3e290021c9e4c78c504fc70fad253c9..991b469d2356d48ed88ecc5aa4256b1a52e42ad5 100644
--- a/vp8/encoder/onyx_if.c
+++ b/vp8/encoder/onyx_if.c
@@ -152,10 +152,6 @@ extern int b_modes[10]  ;
 extern int inter_y_modes[10] ;
 extern int inter_uv_modes[4] ;
 extern unsigned int inter_b_modes[15];
-#if CONFIG_SEGMENTATION
-extern int segment_modes_intra[MAX_MB_SEGMENTS];
-extern int segment_modes_inter[MAX_MB_SEGMENTS];
-#endif
 #endif
 
 extern void (*vp8_short_fdct4x4)(short *input, short *output, int pitch);
@@ -2529,9 +2525,6 @@ void vp8_remove_compressor(VP8_PTR *ptr)
                 fprintf(f, "\n");
 
             }
-#if CONFIG_SEGMENTATION
-            fprintf(f, "Segments:%8d, %8d, %8d, %8d\n", segment_modes_intra[0], segment_modes_intra[1], segment_modes_intra[2], segment_modes_intra[3]);
-#endif
 
             fprintf(f, "Modes in Inter Frames:\n");
             fprintf(f, "Y: %8d, %8d, %8d, %8d, %8d, %8d, %8d, %8d, %8d, %8d\n",
@@ -2550,10 +2543,6 @@ void vp8_remove_compressor(VP8_PTR *ptr)
             }
             fprintf(f, "P:%8d, %8d, %8d, %8d\n", count_mb_seg[0], count_mb_seg[1], count_mb_seg[2], count_mb_seg[3]);
             fprintf(f, "PB:%8d, %8d, %8d, %8d\n", inter_b_modes[LEFT4X4], inter_b_modes[ABOVE4X4], inter_b_modes[ZERO4X4], inter_b_modes[NEW4X4]);
-
-#if CONFIG_SEGMENTATION
-            fprintf(f, "Segments:%8d, %8d, %8d, %8d\n", segment_modes_inter[0], segment_modes_inter[1], segment_modes_inter[2], segment_modes_inter[3]);
-#endif
             fclose(f);
         }
 #endif
diff --git a/vp8/encoder/onyx_int.h b/vp8/encoder/onyx_int.h
index 8bcf65acc073e38fe4fc12f7c2575e5efeb0b6aa..c35dfed087aba49a57a018ea76d1207241b9e1fd 100644
--- a/vp8/encoder/onyx_int.h
+++ b/vp8/encoder/onyx_int.h
@@ -56,12 +56,6 @@
 #define VP8_TEMPORAL_ALT_REF 1
 #endif
 
-#if CONFIG_SEGMENTATION
-#define SEEK_SEGID 12
-#define SEEK_SAMEID 4
-#define SEEK_DIFFID 7
-#endif
-
 typedef struct
 {
     int kf_indicated;
@@ -226,11 +220,7 @@ typedef struct
 typedef struct
 {
     MACROBLOCK  mb;
-#if CONFIG_SEGMENTATION
-    int segment_counts[MAX_MB_SEGMENTS + 8];
-#else
     int segment_counts[MAX_MB_SEGMENTS];
-#endif
     int totalrate;
 } MB_ROW_COMP;
 
diff --git a/vp8/encoder/segmentation.c b/vp8/encoder/segmentation.c
index 16dd74f043fc597e6962d2d5a5123957539573c4..2eb0ce408c9e936f1710b8529833d605ae253413 100644
--- a/vp8/encoder/segmentation.c
+++ b/vp8/encoder/segmentation.c
@@ -116,123 +116,209 @@ void vp8_set_segment_data(VP8_PTR ptr,
 }
 
 #if CONFIG_SEGMENTATION
-void choose_segmap_coding_method( VP8_COMP *cpi,
-                                  int * segment_counts )
+// Based on set of segment counts calculate a probability tree
+void calc_segtree_probs( MACROBLOCKD * xd,
+                         int * segcounts,
+                         vp8_prob * segment_tree_probs )
 {
-    VP8_COMMON *const cm = & cpi->common;
-    MACROBLOCKD *const xd = & cpi->mb.e_mbd;
-
+    int count1,count2;
     int tot_count;
     int i;
-    int count1,count2,count3,count4;
-    int prob[3];
-    int new_cost, original_cost;
 
-    // Select the coding strategy for the segment map (temporal or spatial)
-    tot_count = segment_counts[12] + segment_counts[13] +
-                segment_counts[14] + segment_counts[15];
-    count1 = segment_counts[12] + segment_counts[13];
-    count2 = segment_counts[14] + segment_counts[15];
+    // Blank the strtucture to start with
+    vpx_memset(segment_tree_probs, 0, sizeof(segment_tree_probs));
+
+    // Total count for all segments
+    count1 = segcounts[0] + segcounts[1];
+    count2 = segcounts[2] + segcounts[3];
+    tot_count = count1 + count2;
 
+    // Work out probabilities of each segment
     if (tot_count)
-        prob[0] = (count1 * 255) / tot_count;
+        segment_tree_probs[0] = (count1 * 255) / tot_count;
+    if (count1 > 0)
+        segment_tree_probs[1] = (segcounts[0] * 255) / count1;
+    if (count2 > 0)
+        segment_tree_probs[2] = (segcounts[2] * 255) / count2;
+
+    // Clamp probabilities to minimum allowed value
+    for (i = 0; i < MB_FEATURE_TREE_PROBS; i++)
+    {
+        if (segment_tree_probs[i] == 0)
+            segment_tree_probs[i] = 1;
+    }
+}
+
+// Based on set of segment counts and probabilities calculate a cost estimate
+int cost_segmap( MACROBLOCKD * xd,
+                 int * segcounts,
+                 vp8_prob * probs )
+{
+    int cost;
+    int count1,count2;
+
+    // Cost the top node of the tree
+    count1 = segcounts[0] + segcounts[1];
+    count2 = segcounts[2] + segcounts[3];
+    cost = count1 * vp8_cost_zero(probs[0]) +
+           count2 * vp8_cost_one(probs[0]);
 
+    // Now add the cost of each individual segment branch
     if (count1 > 0)
-        prob[1] = (segment_counts[12] * 255) /count1;
+        cost += segcounts[0] * vp8_cost_zero(probs[1]) +
+                segcounts[1] * vp8_cost_one(probs[1]);
 
     if (count2 > 0)
-        prob[2] = (segment_counts[14] * 255) /count2;
+        cost += segcounts[2] * vp8_cost_zero(probs[2]) +
+                segcounts[3] * vp8_cost_one(probs[2]) ;
 
-    if (cm->frame_type != KEY_FRAME)
-    {
-        tot_count = segment_counts[4] + segment_counts[7];
-        if (tot_count)
-            xd->mb_segment_tree_probs[3] = (segment_counts[4] * 255)/tot_count;
+    return cost;
 
-        tot_count = segment_counts[5] + segment_counts[8];
-        if (tot_count)
-            xd->mb_segment_tree_probs[4] = (segment_counts[5] * 255)/tot_count;
+}
 
-        tot_count = segment_counts[6] + segment_counts[9];
-        if (tot_count)
-            xd->mb_segment_tree_probs[5] = (segment_counts[6] * 255)/tot_count;
-    }
+void choose_segmap_coding_method( VP8_COMP *cpi )
+{
+    VP8_COMMON *const cm = & cpi->common;
+    MACROBLOCKD *const xd = & cpi->mb.e_mbd;
+
+    int i;
+    int tot_count;
+    int no_pred_cost;
+    int t_pred_cost = INT_MAX;
+    int pred_context;
 
-    tot_count = segment_counts[0] + segment_counts[1] +
-                segment_counts[2] + segment_counts[3];
-    count3 = segment_counts[0] + segment_counts[1];
-    count4 = segment_counts[2] + segment_counts[3];
+    int mb_row, mb_col;
+    int segmap_index = 0;
+    unsigned char segment_id;
 
-    if (tot_count)
-        xd->mb_segment_tree_probs[0] = (count3 * 255) / tot_count;
+    int temporal_predictor_count[SEGMENT_PREDICTION_PROBS][2];
+    int no_pred_segcounts[MAX_MB_SEGMENTS];
+    int t_unpred_seg_counts[MAX_MB_SEGMENTS];
 
-    if (count3 > 0)
-        xd->mb_segment_tree_probs[1] = (segment_counts[0] * 255) /count3;
+    vp8_prob no_pred_tree[MB_FEATURE_TREE_PROBS];
+    vp8_prob t_pred_tree[MB_FEATURE_TREE_PROBS];
+    vp8_prob t_nopred_prob[SEGMENT_PREDICTION_PROBS];
 
-    if (count4 > 0)
-        xd->mb_segment_tree_probs[2] = (segment_counts[2] * 255) /count4;
+    vpx_memset(no_pred_segcounts, 0, sizeof(no_pred_segcounts));
+    vpx_memset(t_unpred_seg_counts, 0, sizeof(t_unpred_seg_counts));
+    vpx_memset(temporal_predictor_count, 0, sizeof(temporal_predictor_count));
 
-    for (i = 0; i < MB_FEATURE_TREE_PROBS+3; i++)
+    // First of all generate stats regarding how well the last segment map
+    // predicts this one
+
+    // Initialize macroblod decoder mode info context for to the first mb
+    // in the frame
+    xd->mode_info_context = cm->mi;
+
+    for (mb_row = 0; mb_row < cm->mb_rows; mb_row++)
     {
-        if (xd->mb_segment_tree_probs[i] == 0)
-            xd->mb_segment_tree_probs[i] = 1;
-    }
+        for (mb_col = 0; mb_col < cm->mb_cols; mb_col++)
+        {
+            segment_id = xd->mode_info_context->mbmi.segment_id;
 
-    original_cost = count1 * vp8_cost_zero(prob[0]) +
-                    count2 * vp8_cost_one(prob[0]);
+            // Count the number of hits on each segment with no prediction
+            no_pred_segcounts[segment_id]++;
 
-    if (count1 > 0)
-        original_cost += segment_counts[12] * vp8_cost_zero(prob[1]) +
-                         segment_counts[13] * vp8_cost_one(prob[1]);
+            // Temporal prediction not allowed on key frames
+            if (cm->frame_type != KEY_FRAME)
+            {
+                // Get temporal prediction context
+                pred_context = 0;
+                if (mb_col != 0)
+                    pred_context +=
+                        (xd->mode_info_context-1)->mbmi.segment_flag;
+                if (mb_row != 0)
+                    pred_context +=
+                        (xd->mode_info_context-cm->mb_cols)->mbmi.segment_flag;
+
+                // Test to see if the last frame segment id at the same
+                // locationcorrectly predicts the segment_id for this MB.
+                // Update the prediction flag and count as appropriate;
+                if ( segment_id == cpi->last_segmentation_map[segmap_index] )
+                {
+                    //xd->mode_info_context->mbmi.segment_predicted = 1;
+                    xd->mode_info_context->mbmi.segment_flag = 0;
+                    temporal_predictor_count[pred_context][0]++;
+                }
+                else
+                {
+                    //xd->mode_info_context->mbmi.segment_predicted = 0;
+                    xd->mode_info_context->mbmi.segment_flag = 1;
+                    temporal_predictor_count[pred_context][1]++;
 
-    if (count2 > 0)
-        original_cost += segment_counts[14] * vp8_cost_zero(prob[2]) +
-                         segment_counts[15] * vp8_cost_one(prob[2]) ;
+                    // Update the "undpredicted" segment count
+                    t_unpred_seg_counts[segment_id]++;
+                }
+            }
 
-    new_cost = 0;
+            // Step on to the next mb
+            xd->mode_info_context++;
 
+            // Step on to the next entry in the segment maps
+            segmap_index++;
+        }
+
+        // this is to account for the border in mode_info_context
+        xd->mode_info_context++;
+    }
+
+    // Work out probability tree for coding segments without prediction
+    // and the cost.
+    calc_segtree_probs( xd, no_pred_segcounts, no_pred_tree );
+    no_pred_cost = cost_segmap( xd, no_pred_segcounts, no_pred_tree );
+
+    // Key frames cannot use temporal prediction
     if (cm->frame_type != KEY_FRAME)
     {
-        new_cost = segment_counts[4] *
-                        vp8_cost_zero(xd->mb_segment_tree_probs[3]) +
-                   segment_counts[7] *
-                        vp8_cost_one(xd->mb_segment_tree_probs[3]);
-
-        new_cost += segment_counts[5] *
-                        vp8_cost_zero(xd->mb_segment_tree_probs[4]) +
-                    segment_counts[8] *
-                        vp8_cost_one(xd->mb_segment_tree_probs[4]);
-
-        new_cost += segment_counts[6] *
-                        vp8_cost_zero(xd->mb_segment_tree_probs[5]) +
-                    segment_counts[9] *
-                        vp8_cost_one (xd->mb_segment_tree_probs[5]);
-    }
+        // Work out probability tree for coding those segments not
+        // predicted using the temporal method and the cost.
+        calc_segtree_probs( xd, t_unpred_seg_counts, t_pred_tree );
+        t_pred_cost = cost_segmap( xd, t_unpred_seg_counts, t_pred_tree );
 
-    if (tot_count > 0)
-        new_cost += count3 * vp8_cost_zero(xd->mb_segment_tree_probs[0]) +
-                    count4 * vp8_cost_one(xd->mb_segment_tree_probs[0]);
+        // Add in the cost of the signalling for each prediction context
+        for ( i = 0; i < SEGMENT_PREDICTION_PROBS; i++ )
+        {
+            tot_count = temporal_predictor_count[i][0] +
+                        temporal_predictor_count[i][1];
 
-    if (count3 > 0)
-        new_cost += segment_counts[0] *
-                        vp8_cost_zero(xd->mb_segment_tree_probs[1]) +
-                    segment_counts[1] *
-                        vp8_cost_one(xd->mb_segment_tree_probs[1]);
+            // Work out the context probabilities for the segment
+            // prediction flag
+            if ( tot_count )
+            {
+                t_nopred_prob[i] = ( temporal_predictor_count[i][0] * 255 ) /
+                                   tot_count;
 
-    if (count4 > 0)
-        new_cost += segment_counts[2] *
-                        vp8_cost_zero(xd->mb_segment_tree_probs[2]) +
-                    segment_counts[3] *
-                        vp8_cost_one(xd->mb_segment_tree_probs[2]) ;
+                // Clamp to minimum allowed value
+                if ( t_nopred_prob[i] < 1 )
+                    t_nopred_prob[i] = 1;
+            }
+            else
+                t_nopred_prob[i] = 1;
+
+            // Add in the predictor signaling cost
+            t_pred_cost += ( temporal_predictor_count[i][0] *
+                               vp8_cost_zero(t_nopred_prob[i]) ) +
+                           ( temporal_predictor_count[i][1] *
+                               vp8_cost_one(t_nopred_prob[i]) );
+        }
+    }
 
-    if (new_cost < original_cost)
-        xd->temporal_update = 1;
+    // Now choose which coding method to use.
+    if ( t_pred_cost < no_pred_cost )
+    {
+         xd->temporal_update = 1;
+         vpx_memcpy( xd->mb_segment_tree_probs,
+                     t_pred_tree, sizeof(t_pred_tree) );
+         vpx_memcpy( &xd->mb_segment_pred_probs,
+                     t_nopred_prob, sizeof(t_nopred_prob) );
+    }
     else
     {
-        xd->temporal_update = 0;
-        xd->mb_segment_tree_probs[0] = prob[0];
-        xd->mb_segment_tree_probs[1] = prob[1];
-        xd->mb_segment_tree_probs[2] = prob[2];
+         xd->temporal_update = 0;
+         vpx_memcpy( xd->mb_segment_tree_probs,
+                     no_pred_tree, sizeof(no_pred_tree) );
+         //vpx_memcpy( &xd->mb_segment_pred_probs,
+         //            t_nopred_prob, sizeof(t_nopred_prob) );
     }
 }
 #endif
diff --git a/vp8/encoder/segmentation.h b/vp8/encoder/segmentation.h
index b6394fd095af9c04204e3f81f230397d7679d43c..f3f2b9c6f847ad2de187ad7ed91c8773fa7b837f 100644
--- a/vp8/encoder/segmentation.h
+++ b/vp8/encoder/segmentation.h
@@ -39,8 +39,7 @@ extern void vp8_set_segmentation_map(VP8_PTR ptr, unsigned char *segmentation_ma
 extern void vp8_set_segment_data(VP8_PTR ptr, signed char *feature_data, unsigned char abs_delta);
 
 #if CONFIG_SEGMENTATION
-extern void choose_segmap_coding_method( VP8_COMP *cpi,
-                                         int * segment_counts );
+extern void choose_segmap_coding_method( VP8_COMP *cpi );
 #endif
 
 #endif /* __INC_SEGMENTATION_H__ */