#ifdef CONFIG_HIST_TRIGGERS
        "      hist trigger\t- If set, event hits are aggregated into a hash table\n"
        "\t    Format: hist:keys=<field1>\n"
+       "\t            [:values=<field1[,field2,...]>]\n"
        "\t            [:size=#entries]\n"
        "\t            [if <filter>]\n\n"
        "\t    When a matching event is hit, an entry is added to a hash\n"
-       "\t    table using the key named, and the value of a sum called\n"
-       "\t    'hitcount' is incremented.  Keys correspond to fields in the\n"
-       "\t    event's format description.  Keys can be any field.  The\n"
-       "\t    'size' parameter can be  used to specify more or fewer than\n"
-       "\t    the default 2048 entries for the hashtable size.\n\n"
+       "\t    table using the key(s) and value(s) named, and the value of a\n"
+       "\t    sum called 'hitcount' is incremented.  Keys and values\n"
+       "\t    correspond to fields in the event's format description.  Keys\n"
+       "\t    can be any field.  Values must correspond to numeric fields.\n"
+       "\t    The 'size' parameter can be  used to specify more or fewer\n"
+       "\t    than the default 2048 entries for the hashtable size.\n\n"
        "\t    Reading the 'hist' file for the event will dump the hash\n"
        "\t    table in its entirety to stdout."
 #endif
 
 
 struct hist_trigger_attrs {
        char            *keys_str;
+       char            *vals_str;
        unsigned int    map_bits;
 };
 
                return;
 
        kfree(attrs->keys_str);
+       kfree(attrs->vals_str);
        kfree(attrs);
 }
 
                if ((strncmp(str, "key=", strlen("key=")) == 0) ||
                    (strncmp(str, "keys=", strlen("keys=")) == 0))
                        attrs->keys_str = kstrdup(str, GFP_KERNEL);
+               else if ((strncmp(str, "val=", strlen("val=")) == 0) ||
+                        (strncmp(str, "vals=", strlen("vals=")) == 0) ||
+                        (strncmp(str, "values=", strlen("values=")) == 0))
+                       attrs->vals_str = kstrdup(str, GFP_KERNEL);
                else if (strncmp(str, "size=", strlen("size=")) == 0) {
                        int map_bits = parse_map_size(str);
 
        return 0;
 }
 
+static int create_val_field(struct hist_trigger_data *hist_data,
+                           unsigned int val_idx,
+                           struct trace_event_file *file,
+                           char *field_str)
+{
+       struct ftrace_event_field *field = NULL;
+       unsigned long flags = 0;
+       int ret = 0;
+
+       if (WARN_ON(val_idx >= TRACING_MAP_VALS_MAX))
+               return -EINVAL;
+       field = trace_find_event_field(file->event_call, field_str);
+       if (!field) {
+               ret = -EINVAL;
+               goto out;
+       }
+
+       hist_data->fields[val_idx] = create_hist_field(field, flags);
+       if (!hist_data->fields[val_idx]) {
+               ret = -ENOMEM;
+               goto out;
+       }
+
+       ++hist_data->n_vals;
+
+       if (WARN_ON(hist_data->n_vals > TRACING_MAP_VALS_MAX))
+               ret = -EINVAL;
+ out:
+       return ret;
+}
+
 static int create_val_fields(struct hist_trigger_data *hist_data,
                             struct trace_event_file *file)
 {
+       char *fields_str, *field_str;
+       unsigned int i, j;
        int ret;
 
        ret = create_hitcount_val(hist_data);
+       if (ret)
+               goto out;
 
+       fields_str = hist_data->attrs->vals_str;
+       if (!fields_str)
+               goto out;
+
+       strsep(&fields_str, "=");
+       if (!fields_str)
+               goto out;
+
+       for (i = 0, j = 1; i < TRACING_MAP_VALS_MAX &&
+                    j < TRACING_MAP_VALS_MAX; i++) {
+               field_str = strsep(&fields_str, ",");
+               if (!field_str)
+                       break;
+               if (strcmp(field_str, "hitcount") == 0)
+                       continue;
+               ret = create_val_field(hist_data, j++, file, field_str);
+               if (ret)
+                       goto out;
+       }
+       if (fields_str && (strcmp(fields_str, "hitcount") != 0))
+               ret = -EINVAL;
+ out:
        return ret;
 }
 
        seq_printf(m, " hitcount: %10llu",
                   tracing_map_read_sum(elt, HITCOUNT_IDX));
 
+       for (i = 1; i < hist_data->n_vals; i++) {
+               seq_printf(m, "  %s: %10llu",
+                          hist_data->fields[i]->field->name,
+                          tracing_map_read_sum(elt, i));
+       }
+
        seq_puts(m, "\n");
 }
 
        }
 
        seq_puts(m, ":vals=");
-       seq_puts(m, "hitcount");
+
+       for_each_hist_val_field(i, hist_data) {
+               if (i == HITCOUNT_IDX)
+                       seq_puts(m, "hitcount");
+               else {
+                       seq_puts(m, ",");
+                       hist_field_print(m, hist_data->fields[i]);
+               }
+       }
 
        seq_puts(m, ":sort=");
        seq_puts(m, "hitcount");