Skip site navigation (1)Skip section navigation (2)
Date:      Thu, 28 Apr 2011 01:38:11 +0300
From:      Kostik Belousov <kostikbel@gmail.com>
To:        Bartosz Fabianowski <freebsd@chillt.de>
Cc:        freebsd-hackers@freebsd.org, Hans Petter Selasky <hselasky@c2i.net>
Subject:   Re: Is there some implicit locking of device methods?
Message-ID:  <20110427223811.GA48734@deviant.kiev.zoral.com.ua>
In-Reply-To: <4DB89559.3080008@chillt.de>
References:  <4DB695DB.1080505@chillt.de> <201104271019.31844.jhb@freebsd.org> <4DB818A3.1020104@chillt.de> <201104271334.07170.jhb@freebsd.org> <4DB8873C.5020608@chillt.de> <20110427215228.GY48734@deviant.kiev.zoral.com.ua> <4DB89559.3080008@chillt.de>

next in thread | previous in thread | raw e-mail | index | archive | help

--hImnF3KMhyKrbB+/
Content-Type: text/plain; charset=us-ascii
Content-Disposition: inline
Content-Transfer-Encoding: quoted-printable

On Thu, Apr 28, 2011 at 12:14:49AM +0200, Bartosz Fabianowski wrote:
> Indeed, I may have mixed up terminology. Sorry about that. What I am=20
> doing (or trying to do) is very simple:
>=20
> There is a single physical USB device. I have a single device node=20
> representing it. This device can be opened for reading, concurrently,=20
> any number of times. Everyone who open()s the device can read() it at=20
> their own pace. I implemented this by maintaining an individual queue of=
=20
> incoming data for each open() call. This queue resides in cdevpriv.
>=20
> So open() instantiates a queue and adds it to the driver's global list=20
> of queues. Whenever a packet arrives from the device, it is placed in=20
> all the queues (I have a linked list of all queues for that purpose).=20
> When the open() is eventually followed by a close(), the cdevpriv=20
> destructor removes the queue from the global list and frees its memory.
>=20
> In addition to this, I need to start the USB transfer when the first=20
> open() occurs and stop it again when the last close() occurs. I am doing=
=20
> this by checking the length of the global list. When the list is=20
> zero-length on open(), I start the transfer. When the list i zero-length=
=20
> in the cdevpriv destructor, I stop the transfer.
>=20
> I cannot see how else to achieve this behavior (other than device=20
> cloning which I was using before but which is more complicated and=20
> probably more error-prone). If I am doing something wrong and there is a=
=20
> more correct way to do it, I would love to hear about it.

This is a strange architecture, esp. amusing is the kernel-mode
traffic multiplier. Without knowing the details, and really not wanting
to know it, I could make two suggestions out of thin air:

- use usermode daemon that multiplies traffic for all connected clients;

- or, implement a ring buffer that cyclically stores the received data,
  and keep only the current read pointer in the cdevpriv. You need to
  handle the overflow case (eq. to the stuck reader) somehow in the
  current scheme anyway. Reader may now read from its current read
  position in the buffer up to the fill point. If the buffer wrapped
  for the reader, it should get some error.

--hImnF3KMhyKrbB+/
Content-Type: application/pgp-signature
Content-Disposition: inline

-----BEGIN PGP SIGNATURE-----
Version: GnuPG v1.4.11 (FreeBSD)

iEYEARECAAYFAk24mtIACgkQC3+MBN1Mb4i5GQCfV+389ZZ9qk3HcxpdKFPCu0Uo
dYkAn0kMKW/F+xqu2KcWKoefadqjNU1Y
=P07P
-----END PGP SIGNATURE-----

--hImnF3KMhyKrbB+/--



Want to link to this message? Use this URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?20110427223811.GA48734>