summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorKenjiro Cho <kjc@cvs.openbsd.org>2002-05-17 07:19:45 +0000
committerKenjiro Cho <kjc@cvs.openbsd.org>2002-05-17 07:19:45 +0000
commit483ceccac7d0851361ab279215972e0c29cb6223 (patch)
tree53ad3b3425e766c62201fa56482b9a1414d25c18
parent66237872139a844457e646fc63357f62a6b17281 (diff)
sync with KAME.
- don't MALLOC() with M_WAITOK in the spl block. move the allocation before splimp(). - when we reset vt of a class, reset the runtime service curve as well. - don't use max() to compare 64 bit values.
-rw-r--r--sys/altq/altq_hfsc.c76
1 files changed, 40 insertions, 36 deletions
diff --git a/sys/altq/altq_hfsc.c b/sys/altq/altq_hfsc.c
index 75b9ad11f64..98d436bec99 100644
--- a/sys/altq/altq_hfsc.c
+++ b/sys/altq/altq_hfsc.c
@@ -1,5 +1,5 @@
-/* $OpenBSD: altq_hfsc.c,v 1.4 2002/03/14 01:26:26 millert Exp $ */
-/* $KAME: altq_hfsc.c,v 1.8 2000/12/14 08:12:46 thorpej Exp $ */
+/* $OpenBSD: altq_hfsc.c,v 1.5 2002/05/17 07:19:44 kjc Exp $ */
+/* $KAME: altq_hfsc.c,v 1.13 2002/05/16 11:02:58 kjc Exp $ */
/*
* Copyright (c) 1997-1999 Carnegie Mellon University. All Rights Reserved.
@@ -70,11 +70,10 @@ static struct hfsc_class *hfsc_class_create(struct hfsc_if *,
struct service_curve *, struct hfsc_class *, int, int);
static int hfsc_class_destroy(struct hfsc_class *);
static int hfsc_class_modify(struct hfsc_class *,
- struct service_curve *, struct service_curve *);
+ struct service_curve *, struct service_curve *);
static struct hfsc_class *hfsc_nextclass(struct hfsc_class *);
-static int hfsc_enqueue(struct ifaltq *, struct mbuf *,
- struct altq_pktattr *);
+static int hfsc_enqueue(struct ifaltq *, struct mbuf *, struct altq_pktattr *);
static struct mbuf *hfsc_dequeue(struct ifaltq *, int);
static int hfsc_addq(struct hfsc_class *, struct mbuf *);
@@ -112,11 +111,11 @@ static u_int dx2d(u_int64_t);
static void sc2isc(struct service_curve *, struct internal_sc *);
static void rtsc_init(struct runtime_sc *, struct internal_sc *,
- u_int64_t, u_int64_t);
+ u_int64_t, u_int64_t);
static u_int64_t rtsc_y2x(struct runtime_sc *, u_int64_t);
static u_int64_t rtsc_x2y(struct runtime_sc *, u_int64_t);
static void rtsc_min(struct runtime_sc *, struct internal_sc *,
- u_int64_t, u_int64_t);
+ u_int64_t, u_int64_t);
int hfscopen(dev_t, int, int, struct proc *);
int hfscclose(dev_t, int, int, struct proc *);
@@ -471,9 +470,24 @@ hfsc_class_modify(cl, rsc, fsc)
struct hfsc_class *cl;
struct service_curve *rsc, *fsc;
{
- struct internal_sc *tmp;
+ struct internal_sc *rsc_tmp, *fsc_tmp;
int s;
+ if (rsc != NULL && (rsc->m1 != 0 || rsc->m2 != 0) &&
+ cl->cl_rsc == NULL) {
+ MALLOC(rsc_tmp, struct internal_sc *,
+ sizeof(struct internal_sc), M_DEVBUF, M_WAITOK);
+ if (rsc_tmp == NULL)
+ return (ENOMEM);
+ }
+ if (fsc != NULL && (fsc->m1 != 0 || fsc->m2 != 0) &&
+ cl->cl_fsc == NULL) {
+ MALLOC(fsc_tmp, struct internal_sc *,
+ sizeof(struct internal_sc), M_DEVBUF, M_WAITOK);
+ if (fsc_tmp == NULL)
+ return (ENOMEM);
+ }
+
s = splimp();
if (!qempty(cl->cl_q))
hfsc_purgeq(cl);
@@ -485,16 +499,8 @@ hfsc_class_modify(cl, rsc, fsc)
cl->cl_rsc = NULL;
}
} else {
- if (cl->cl_rsc == NULL) {
- MALLOC(tmp, struct internal_sc *,
- sizeof(struct internal_sc),
- M_DEVBUF, M_WAITOK);
- if (tmp == NULL) {
- splx(s);
- return (ENOMEM);
- }
- cl->cl_rsc = tmp;
- }
+ if (cl->cl_rsc == NULL)
+ cl->cl_rsc = rsc_tmp;
bzero(cl->cl_rsc, sizeof(struct internal_sc));
sc2isc(rsc, cl->cl_rsc);
rtsc_init(&cl->cl_deadline, cl->cl_rsc, 0, 0);
@@ -509,16 +515,8 @@ hfsc_class_modify(cl, rsc, fsc)
cl->cl_fsc = NULL;
}
} else {
- if (cl->cl_fsc == NULL) {
- MALLOC(tmp, struct internal_sc *,
- sizeof(struct internal_sc),
- M_DEVBUF, M_WAITOK);
- if (tmp == NULL) {
- splx(s);
- return (ENOMEM);
- }
- cl->cl_fsc = tmp;
- }
+ if (cl->cl_fsc == NULL)
+ cl->cl_fsc = fsc_tmp;
bzero(cl->cl_fsc, sizeof(struct internal_sc));
sc2isc(fsc, cl->cl_fsc);
rtsc_init(&cl->cl_virtual, cl->cl_fsc, 0, 0);
@@ -851,6 +849,16 @@ init_v(cl, len)
/* already active */
break;
+ /*
+ * if parent became idle while this class was idle.
+ * reset vt and the runtime service curve.
+ */
+ if (cl->cl_parent->cl_nactive == 0 ||
+ cl->cl_parent->cl_vtperiod != cl->cl_parentperiod) {
+ cl->cl_vt = 0;
+ rtsc_init(&cl->cl_virtual, cl->cl_fsc,
+ 0, cl->cl_total);
+ }
min_cl = actlist_first(cl->cl_parent->cl_actc);
if (min_cl != NULL) {
u_int64_t vt;
@@ -862,17 +870,13 @@ init_v(cl, len)
*/
max_cl = actlist_last(cl->cl_parent->cl_actc);
vt = (min_cl->cl_vt + max_cl->cl_vt) / 2;
- if (cl->cl_parent->cl_vtperiod == cl->cl_parentperiod)
- vt = max(cl->cl_vt, vt);
- cl->cl_vt = vt;
- } else {
- /* no packet is backlogged. set vt to 0 */
- cl->cl_vt = 0;
+ if (cl->cl_parent->cl_vtperiod != cl->cl_parentperiod
+ || vt > cl->cl_vt)
+ cl->cl_vt = vt;
}
/* update the virtual curve */
- rtsc_min(&cl->cl_virtual, cl->cl_fsc,
- cl->cl_vt, cl->cl_total);
+ rtsc_min(&cl->cl_virtual, cl->cl_fsc, cl->cl_vt, cl->cl_total);
cl->cl_vtperiod++; /* increment vt period */
cl->cl_parentperiod = cl->cl_parent->cl_vtperiod;